9

如果有人能解释一下,我将不胜感激,因为我对这个问题有点无能为力。基本上,我正在尝试使用构建在rails-api gem之上的仅 JSON rails rest api 应用程序来完成设计身份验证。

我已经实现了SessionsRegistrations处理,如下所述。

class ApplicationController < ActionController::API
    include ActionController::MimeResponds
end

session_controller.rb

  class SessionsController < Devise::SessionsController
    prepend_before_filter :require_no_authentication, :only => [:create ]

    before_filter :ensure_params_exist
    def create
      build_resource
      resource = User.find_for_database_authentication(:email => params[:user][:email])
      return invalid_login_attempt unless resource

      if resource.valid_password?(params[:user][:password])
        sign_in("user", resource)
        render :json=> {:success=>true, :auth_token=>resource.authentication_token, :email=>resource.email}
      return
      end
      invalid_login_attempt
    end

    def destroy
      sign_out(resource_name)
    end

    protected

    def ensure_params_exist
      return unless params[:user][:email].blank?
      render :json=>{:success=>false, :message=>"missing login email parameter"}, :status=>422
    end

    def invalid_login_attempt
      warden.custom_failure!
      render :json=> {:success=>false, :message=>"Error with your login or password"}, :status=>401
    end

  end

注册控制器.rb

  class RegistrationsController < Devise::RegistrationsController
    skip_before_filter :verify_authenticity_token, :only => :create

    def create
      user = User.new(params[:user])
      if user.save
        render :json=> {:user => [:email => user.email, :auth_token => user.authentication_token]}, :status => 201
      return
      else
        warden.custom_failure!
        render :json=> user.errors, :status=>422
      end
    end
  end

到目前为止一切正常。在我的其他控制器中,我添加了这条线来保护它们。before_filter :authenticate_user!但是使用 auth_token 调用时出现此错误。?auth_token=xxxxxxxxxxx

undefined method authenticate_user!

控制器
class RestaurantsController < ApplicationController

        before_filter :authenticate_user!

      def index
        @restaurants =  Restaurant.all

      end

      def show
        @restaurant = Restaurant.find(params[:id])

      end
    end

仍然不确定这里可能是什么问题 - 我假设这种情况正在发生,因为在使用时缺少正确设计的东西ActionController::API

更新 似乎问题出在我routes.rb自己身上 - 这就是路线的完成方式。

require 'api_constraints'

MyApi::Application.routes.draw do

  namespace :api, defaults: {format: 'json'} do
    scope module: :v1, constraints: ApiConstraints.new(version: 1,default: true) do
      devise_for :users
      resources :friends
      resources :locations
      resources :profiles
      resources :users


    end

    scope module: :v2, constraints: ApiConstraints.new(version: 2) do
    # Future releases of the API can go here
    end

  end

end

现在,如果在范围外重复,devise_for :users一切都开始工作了。

require 'api_constraints'

MyApi::Application.routes.draw do

  namespace :api, defaults: {format: 'json'} do
    scope module: :v1, constraints: ApiConstraints.new(version: 1,default: true) do
      devise_for :users
      resources :friends
      resources :locations
      resources :profiles
      resources :users


    end

    scope module: :v2, constraints: ApiConstraints.new(version: 2) do
    # Future releases of the API can go here
    end

  end
devise_for :users
end

有没有人解释为什么?

4

2 回答 2

6

步骤 1. 使用自定义控制器覆盖设计控制器,将重定向替换为 JSON 响应。这是一个使用 JSON 响应的自定义 SessionController

第 2 步。如果发生身份验证错误,则控制权转到使用 failure_app 的守望者,这不过是一个 Rack 应用程序,它设置闪存消息并根据请求的格式呈现/重定向。您需要一个自定义 json 响应,其中包含一个包含错误数组的错误键,而不是默认的错误键。所以你需要一个 custom_auth_failure_app.rb 在 config/initializers 下有这个内容

第 4 步。现在我们必须告诉 Devise 使用自定义故障应用程序,方法是将其添加到config/initializers/devise.rb

config.warden do |manager|
  manager.failure_app = CustomAuthFailure
end

步骤 5. 在设计模型中启用token_authenticatable并将以下内容添加到config/initializers/devise.rb

config.http_authenticatable = true
config.skip_session_storage = [:http_auth, :token_auth]

第 6 步。如果您使用的是不使用会话的真正 JSON API,请使用具有处理零会话的补丁的 Warden (> 1.2.2) 版本。

另请阅读我关于使用 Rails4 + Rails-API + Devise 创建经过测试、记录和版本化的 JSON API 的博客文章,其中讨论了所有这些步骤。

于 2013-08-16T08:05:01.563 回答
1

对于遇到同样问题的任何人,例如在使用andauthenticate_user时设计辅助方法不起作用,请尝试将此行添加到 app/controllers/application_controller.rb:devise_token_authrails-api

include DeviseTokenAuth::Concerns::SetUserByToken

当您运行devise_token_auth安装时,它应该会在 ApplicationController 中自动创建此关注点。这种关注提供了对辅助方法的访问权限,例如authenticate_user. 问题是,如果您使用而不是香草导轨,则不会增加这种担忧。rails-api因此,您必须手动添加它。

不确定根本原因,但我怀疑是因为 rails-api 的 ApplicationController 继承自 ActionController::API,这与 vanilla rails 的 ApplicationController 不同,后者继承自 ActionController::Base。

于 2015-08-19T23:34:25.683 回答