如何使用 Devise 根据用户的角色重定向用户的主(根)路径?

发布于 2024-10-13 06:32:06 字数 304 浏览 8 评论 0原文

我正在开发一个项目管理应用程序,在该应用程序中,我有 project_managers客户。我正在使用 Devise 和 CanCan 进行身份验证/授权。

登录后什么时候我应该将用户重定向到他们自己的特定控制器/布局/视图?有没有办法检查 routes.rb 中的 current_user.role 并根据他们是项目经理还是客户来设置根(或重定向) ?这是我可以在 Devise 中进行的更改吗?

预先感谢您的任何帮助! - 标记

I'm working on a project management app, and in the app, I have project_managers and clients. I'm using Devise and CanCan for authentication/authorization.

At what point after login should I be redirecting the user to their own specific controller/layout/views? Is there a way to check for current_user.role in routes.rb and set the root (or redirect) based on whether or not they're a project manager or a client? Is this a change I can make in Devise somewhere?

Thanks in advance for any help!
--Mark

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(5

浮生未歇 2024-10-20 06:32:06

您的 routes.rb 文件不会知道用户具有什么角色,因此您将无法使用它来分配特定的根路由。

您可以做的是设置一个控制器(例如,passthrough_controller.rb),该控制器可以读取角色并重定向。像这样:

# passthrough_controller.rb
class PassthroughController < ApplicationController
  def index
    path = case current_user.role
      when 'project_manager'
        some_path
      when 'client'
        some_other_path
      else
        # If you want to raise an exception or have a default root for users without roles
    end

    redirect_to path     
  end
end

# routes.rb
root :to => 'passthrough#index'

这样,所有用户都将拥有一个入口点,该入口点又根据他们的角色将他们重定向到适当的控制器/操作。

Your routes.rb file won't have any idea what role the user has, so you won't be able to use it to assign specific root routes.

What you can do is set up a controller (for example, passthrough_controller.rb) which in turn can read the role and redirect. Something like this:

# passthrough_controller.rb
class PassthroughController < ApplicationController
  def index
    path = case current_user.role
      when 'project_manager'
        some_path
      when 'client'
        some_other_path
      else
        # If you want to raise an exception or have a default root for users without roles
    end

    redirect_to path     
  end
end

# routes.rb
root :to => 'passthrough#index'

This way, all users will have one point of entry, which in turn redirects them to the appropriate controller/action depending on their role.

﹏雨一样淡蓝的深情 2024-10-20 06:32:06

另一种选择是将过程传递给经过身份验证的方法,如下所示(我在本例中使用 rolify):

authenticated :user, ->(u) { u.has_role?(:manager) } do
  root to: "managers#index", as: :manager_root
end

authenticated :user, ->(u) { u.has_role?(:employee) } do
  root to: "employees#index", as: :employee_root
end

root to: "landing_page#index"

请注意,在 Rails 4 中,您必须指定一个唯一的名称对于每个根路由,请参阅此问题了解详细信息。

Another option is to pass a proc to the authenticated method like this (I'm using rolify in this example):

authenticated :user, ->(u) { u.has_role?(:manager) } do
  root to: "managers#index", as: :manager_root
end

authenticated :user, ->(u) { u.has_role?(:employee) } do
  root to: "employees#index", as: :employee_root
end

root to: "landing_page#index"

Note that in Rails 4 you have to specify a unique name for each root route, see this issue for details.

空心空情空意 2024-10-20 06:32:06

最简单的解决方案是使用 lambda:

root :to => 'project_managers#index', :constraints => lambda { |request| request.env['warden'].user.role == 'project_manager' }
root :to => 'clients#index'

The simplest solution is to use lambda:

root :to => 'project_managers#index', :constraints => lambda { |request| request.env['warden'].user.role == 'project_manager' }
root :to => 'clients#index'
べ映画 2024-10-20 06:32:06

我在使用 Warden 的 Rails 3 应用程序中执行此操作。由于 Devise 是建立在 Warden 之上的,我认为它会为您工作,但在依赖它之前一定要先尝试一下。

class ProjectManagerChecker
  def self.matches?(request)
    request.env['warden'].user.role == 'project_manager'
  end
end

# routes.rb
get  '/' => 'project_managers#index', :constraints => ProjectManagerChecker
get  '/' => 'clients#index'

如果用户的角色是“project_manager”,则将使用 ProjectManagersController - 如果不是,则将使用 ClientsController。如果他们根本没有登录, env['warden'].user 将为 nil,并且您会收到错误,因此您可能需要解决这个问题,但这会让你开始吧。

I do this in a Rails 3 app that uses Warden. Since Devise is built on top of Warden, I think it'll work for you, but be sure to experiment with it a bit before relying on it.

class ProjectManagerChecker
  def self.matches?(request)
    request.env['warden'].user.role == 'project_manager'
  end
end

# routes.rb
get  '/' => 'project_managers#index', :constraints => ProjectManagerChecker
get  '/' => 'clients#index'

If the user's role is "project_manager" the ProjectManagersController will be used - if it's not the ClientsController will be used. If they're not logged in at all, env['warden'].user will be nil and you'll get an error, so you'll probably want to work around that, but this will get you started.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文