Heroku + Devise 1.2 + OmniAuth to GitHub = нулевое значение в столбце «электронная почта» - PullRequest
1 голос
/ 07 февраля 2011

Я пытаюсь настроить Devise 1.2, чтобы разрешить аутентификацию пользователя через GitGub. Насколько я могу проверить его локально, используя Cucumber и отключив GitHub OAuth, похоже, что он работает нормально. Однако после развертывания в Heroku и попытки аутентификации я получаю сообщение об ошибке после перенаправления обратно в свое приложение из GitHub.

Проверяя журнал Heroku, я вижу ошибку ...

PGError: ОШИБКА: нулевое значение в столбце «электронная почта» нарушает ненулевое ограничение

Итак, очевидно, я либо не получаю адрес электронной почты от GitHub, либо он теряется где-то по пути?

Я следовал инструкциям и примерам https://github.com/plataformatec/devise/wiki/OmniAuth:-Overview и https://github.com/plataformatec/devise/wiki/OmniAuth:--Testing-%27facebook%27-signup--%5BRails-3---Cucumber---Capybara---Mongoid-%5D,, но изменил их для GitHub вместо Facebook, ссылаясь на исходный код OmniAuth для стратегии GitHub по адресу https://github.com/intridea/omniauth/blob/master/oa-oauth/lib/omniauth/strategies/github.rb.

Вот что у меня есть для моего файла devise_steps.rb, что приводит к прохождению функции Cucumber:

ACCESS_TOKEN = {
  :access_token => "stevejdev"
}

# Not all pieces of this hash are yet proven to be correct.
# It does, at least supply the necessary information for a
# successful login simulation though.
GITHUB_INFO = {
  :user => {
    :id => '12345', 
    :email => 'johndoe@example.com',
    # login value maps to nickname and to <user> part of
    # "http://github.com/<user>" in urls[GitHub]
    :login => 'johnxd',
    :name => 'John Doe',
    # blog value maps to urls[blog]
    :blog => 'http://blaagstop.com/johndoe',
  }
}

When /^GitHub replies$/ do 
  Devise::OmniAuth.short_circuit_authorizers!
  Devise::OmniAuth.stub!(:github) do |b|
    b.post('/login/oauth/access_token') { 
              [200, {}, ACCESS_TOKEN.to_json] }
    b.get('/api/v2/json/user/show?access_token=stevejdev') {
              [200, {}, GITHUB_INFO.to_json ] }
  end

  visit user_omniauth_callback_path(:github)
end

Вот мой обработчик обратного вызова:

class Users::OmniauthCallbacksController < Devise::OmniauthCallbacksController

  def github
    @user = User.find_for_github_oauth(env["omniauth.auth"], current_user)

    if @user.persisted?
      flash[:notice] = I18n.t(
        "devise.omniauth_callbacks.success", :kind => "GitHub" )
      sign_in_and_redirect @user, :event => :authentication
    else
      session["devise.github_data"] = env["omniauth.auth"]
      redirect_to new_user_registration_url
    end
  end

end

Вот определение find_for_github_oauth в моей модели User:

def self.find_for_github_oauth(access_token, signed_in_resource=nil)
  data = access_token['extra']['user_hash']

  # Find the existing user or create a new one.
  #  Omit the password-generation code shown in the example at
  #  https://github.com/plataformatec/devise/wiki/OmniAuth:-Overview
  #  because we're intending to use OmniAuth only, so
  #  presumably don't need a password.
  User.find_by_email( data["email"] ) ||
  User.create!( :email => data["email"] )

end

Ответы [ 2 ]

0 голосов
/ 24 февраля 2013

Для потомков:

Эта проблема была решена в репозитории omniauth-github gem, в это слияние .

Он недоступен в последней версии, поэтому, если вы хотите иметь его, вам нужно получить самую передовую версию репо, добавив адрес репо в свой gemfile:

gem 'omniauth-github', :git => "git://github.com/intridea/omniauth-github.git"
0 голосов
/ 06 марта 2011

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

Поскольку в будущем я планирую поддерживать несколько провайдеров, я теперь использую комбинацию полей auth_provider и login в качестве логического ключа для таблицы users, и я указал config.authentication_keys = [ :auth_provider, :login ] в файле config / devise.rb. Когда я добавляю поддержку другого провайдера, который использует адрес электронной почты для входа в систему, это нормально. Я просто буду использовать значение адреса электронной почты из ответа поставщика в качестве значения для входа в систему для модели.

...