1

Omniauth gem をインストールしました。現在、Devise で実装されています。

Devise はうまく機能していますが、Omniauth のセットアップは大変でした。Twitter認証でOmniauthを使用しています。

問題: [Twitter でログイン] アイコンをクリックすると、必要に応じて Twitter にリダイレクトされ、Twitter の資格情報を入力するように求められます。リダイレクト (コールバック) が開始されるまでは問題ありません。

アプリにリダイレクトしようとすると、次のエラーが発生します。

AuthenticationsController#create の NoMethodError

undefined method `[]' for nil:NilClass

app/models/user.rb:14:in `apply_omniauth'
app/controllers/authentications_controller.rb:14:in `create'

登録コントローラー:

class RegistrationsController < Devise::RegistrationsController
  def create
    super
    session[:omniauth] = nil unless @user.new_record?
  end

  private

  def build_resource(*args)
    super
    if session[:omniauth]
      @user.apply_omniauth(session[:omniauth])
      @user.valid?
    end
  end
end

認証コントローラー:

class AuthenticationsController < ApplicationController
  def create
    omniauth = request.env["omniauth.auth"]
    authentication = Authentication.find_by_provider_and_uid(omniauth['provider'], omniauth['uid'])
    if authentication
      flash[:notice] = "Signed in successfully."
      sign_in_and_redirect(:user, authentication.user)
    elsif current_user
      current_user.authentications.create!(:provider => omniauth['provider'], :uid => omniauth['uid'])
      flash[:notice] = "Authentication successful."
      redirect_to authentications_url
    else
      user = User.new
      user.apply_omniauth(omniauth)
      if user.save
        flash[:notice] = "Signed in successfully."
        sign_in_and_redirect(:user, user)
      else
        session[:omniauth] = omniauth.except('extra')
        redirect_to new_user_registration_url
      end
    end
  end
end

User.rb (ユーザーコントローラー)

class User < ActiveRecord::Base
  has_many :authentications

  # Include default devise modules. Others available are:
  # :token_authenticatable, :lockable, :timeoutable and :activatable
  devise :database_authenticatable, :registerable, 
         :recoverable, :rememberable, :trackable, :validatable

  # Setup accessible (or protected) attributes for your model
  attr_accessible :email, :password, :password_confirmation
  # attr_accessible :title, :body

  def apply_omniauth(omniauth)
    self.email = omniauth['user_info']['email'] if email.blank?
    authentications.build(:provider => omniauth['provider'], :uid => omniauth['uid'])

  end

  def password_required?
    (authentications.empty? || !password.blank?) && super
  end
end

更新をお読みください:

次のRailsCastに出くわしました。

チュートリアルでは、次を実行するように指示されています。

rails g nifty:scaffold authentication user_id:integer \ provider:string uid:string index create destroy

しかし、私のマシンには気の利いた足場がなく、ただ走った

rails g scaffold 認証 user_id:integer \ provider:string uid:string index create destroy

これは動作が異なります。スタブの「index」、「create」、および「destroy」コントローラー メソッドを作成する代わりに、データベースにフィールドを作成しました。

フィールドを削除するにはどうすればよいですか?

4

2 に答える 2

1

交換してみる

omniauth['user_info']['email']

omniauth['email']

このようなラッパーメソッドを使用できます

更新: omniauth のラッパー

def omni_conversion(omniauth)
    {
      # Required For Social Network Creation
      access_token: omniauth.credentials.token,
      link: omniauth.extra.raw_info.link,
      provider: omniauth.provider,
      providerid: omniauth.uid,

      # Required For User Creation
      birthday: omniauth.extra.raw_info.birthday,
      email: omniauth.info.email,
      first_name: omniauth.info.first_name,
      gender: omniauth.extra.raw_info.gender,
      last_name: omniauth.info.last_name,
      middle_name: omniauth.info.middle_name,
      picture: omniauth.info.image,
      timezone: omniauth.info.timezone,
      username: omniauth.extra.raw_info.username
     }
  end

omniauth = omni_conversion(omniauth)

利点は、シンボルを直接使用できることです。モデルに直接渡すことができます。

于 2013-02-28T06:53:01.150 に答える
0

あなたのUser.rb行#14で

代わりomniauth['user_info']['email']omniauth['info']['email']

気の利いたジェネレーターの場合は、

gem "nifty-generators", group: :development

あなたの中のGemfile

rails g scaffold --help足場ジェネレーターの利用可能なオプションを確認してみてください

また、この驚くべき rails-cast too generators-in-rails-3もご覧ください。

于 2013-02-28T06:59:10.067 に答える