繁体   English   中英

Rails 5,Devise,Omniauth,Twitter

[英]Rails 5, Devise, Omniauth, Twitter

我知道这已被多次询问,但答案对我来说永远不会完全接受。

因此,我正在关注Ryan Bates关于这个主题的Railscast ,并将其与官方的Devise Omniauth指南 (基于FB)混合,但我只是没有像我期望的那样让它工作,所以我希望得到一些帮助。

我有一个Users::OmniauthCallbacksController ,如下所示:

class Users::OmniauthCallbacksController < Devise::OmniauthCallbacksController
  def all
    @user = User.from_omniauth(request.env["omniauth.auth"])

    if @user.persisted?
      sign_in_and_redirect root_path, :event => :authentication #this will throw if @user is not activated
      set_flash_message(:notice, :success, :kind => "Twitter") if is_navigational_format?
    else
      session["devise.twitter_data"] = request.env["omniauth.auth"].except("extra")
      flash[:notice] = flash[:notice].to_a.concat resource.errors.full_messages
      redirect_to new_user_registration_url
    end
  end

  alias_method :twitter, :all

  def failure
    redirect_to root_path
  end
end

然后我的User.rb上也有两个方法

  def self.from_omniauth(auth)
    where(provider: auth.provider, uid: auth.uid).first_or_create do |user|
      user.update(
        email: auth.info.email,
        password: Devise.friendly_token[0,20],
        username: auth.info.nickname,
        remote_avatar_url: auth.info.image,
        token: auth.credentials.token,
        secret: auth.credentials.secret
      )
    end
  end

  def self.new_with_session(params, session)
    super.tap do |user|
      if data = session["devise.twitter_data"]
        # user.attributes = params
        user.update(
          email: params[:email],
          password: Devise.friendly_token[0,20],
          username: data["info"]["nickname"],
          remote_avatar_url: data["info"]["image"],
          token: data["credentials"]["token"],
          secret: data["credentials"]["secret"]
        )
      end
    end
  end

我遇到了各种各样的问题。 最直接的是因为我设置了密码,用户在尝试登录时不知道密码(我确认时不会自动登录)。

但是,如果我没有设置密码,它不会要求他们设置密码......所以这有点奇怪。

这些是我的User模型上的设计设置:

  devise :database_authenticatable, :registerable,
         :recoverable, :rememberable, :trackable, :validatable,
         :confirmable, :omniauthable, :omniauth_providers => [:twitter]

  validates :username,
    presence: true,
    uniqueness: {
      case_sensitive: false
    }

  validate :validate_username

  def validate_username
    if User.where(email: username).exists?
      errors.add(:username, :invalid)
    end
  end

所以我的问题是,当有人通过Twitter注册时,他们是否需要输入密码? 无论如何,我会自动将它们发送到registration/new.html.erb ,因为Twitter不会返回电子邮件值。 但我想在优化之前先让流程先运行。

我该如何处理密码问题?

编辑1

为了更清楚起见,无论OAuth提供商如何,我都必须处理此password_required问题。

那么如何覆盖所有OAuth提供商的要求呢?

您应该将以下方法添加到User类:

def password_required?
  (provider.blank? || uid.blank?) && super
end

由于Twitter不会返回用户的电子邮件,您可能还想调整该电子邮件验证 ,但将用户重定向到您正在进行的registration/new.html.erb似乎对我来说是正确的方法。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM