Pundit 政策的嵌套资源

Posted

技术标签:

【中文标题】Pundit 政策的嵌套资源【英文标题】:Nested Resources For Pundit Policy 【发布时间】:2017-09-19 03:57:54 【问题描述】:

我仍在努力理解 Pundit 的政策。我想我已经很接近了,但我浪费了太多时间试图弄清楚这一点。我的帖子政策效果很好,但尝试授权 cmets,我收到未定义的错误...

cmets_controller.rb

class CommentsController < ApplicationController
before_action :find_comment, only: [:show, :edit, :update, :destroy]
before_action :authenticate_user!, except: [:index, :show]

def create
    @post = Post.find(params[:post_id])
    @comment = @post.comments.create(params[:comment].permit(:comment))
    @comment.user_id = current_user.id if current_user
authorize @comment
    @comment.save

    if @comment.save
        redirect_to post_path(@post)
    else
        render 'new'
    end
end

def edit
authorize @comment
end

def update
authorize @comment
    if @comment.update(params[:comment].permit(:comment))
        redirect_to post_path(@post)
    else
        render 'edit'
    end
end

def destroy
authorize @comment
@comment.destroy
redirect_to post_path(@post)
end

private

  def find_comment
    @post = Post.find(params[:post_id])
    @comment = @post.comments.find(params[:id])
  end
 end

comment_policy.rb

class CommentPolicy < ApplicationPolicy

  def owned
    comment.user_id == user.id
  end

  def create?
    comment.user_id = user.id
    new?
  end

  def new?
    true
  end

  def update?
   edit?
  end

  def edit?
    owned
  end

  def destroy?
    owned
  end
end

格式和缩进有点不对...我发誓我不是这样编码的

  class ApplicationPolicy
    attr_reader :user, :post

    def initialize(user, post)
      raise Pundit::NotAuthorizedError, "must be logged in" unless user
      @user = user
      @post = post
    end

    def index?
      false
    end

    def show?
      scope.where(:id => post.id).exists?
    end

    def create?
      false
    end

    def new?
      create?
    end

    def update?
      false
    end

    def edit?
      update?
    end

    def destroy?
      false
    end

    def scope
      Pundit.policy_scope!(user, post.class)
    end

    class Scope
      attr_reader :user, :scope

      def initialize(user, scope)
        @user = user
        @scope = scope
      end

      def resolve
        scope
      end
    end
  end

【问题讨论】:

你能显示你的错误详情以及它发生在哪里吗? 我认为您应该在您的政策中使用 record 而不是 comment 不确定您的意思。我没有“记录”模型,只有“评论、用户和帖子” 您继承自 ApplicationPolicy。它将您的对象实例化为record。请注意,您没有定义#initialize?因为您使用的是超类'。 尝试将`comment.user_id`更改为`record.user_id` 【参考方案1】:

您在 ApplicationPolicy 中将资源初始化为 @post。由于您的 CommentPolicy 继承自 ApplicationPolicy 并使用其初始化,因此它只能访问 @post。最好的选择是将其保留为record:

  class ApplicationPolicy
    attr_reader :user, :record

    def initialize(user, record)
      raise Pundit::NotAuthorizedError, "must be logged in" unless user
      @user = user
      @record = record
    end
    ## code omitted
  end

  class CommentPolicy < ApplicationPolicy
    def owned
      record.user_id == user.id
    end
    ## code omitted
  end

基本上,您可以随意称呼它,但record 更有意义,因为它将用于不同的策略子类。

【讨论】:

在我完全复制了您所拥有的内容后,仍然出现相同的“未定义”错误。 Post 在 PostPolicy 类中未定义。以防万一,用户有很多帖子,帖子有很多cmet。 您还需要将 PostPolicy 中对 post 的引用更改为 record。在所有策略类中将资源称为 record 已修复,谢谢! 当然。很高兴它有帮助。

以上是关于Pundit 政策的嵌套资源的主要内容,如果未能解决你的问题,请参考以下文章

Rails 4 - pundit - 如何编写 if 语句来检查用户权限

Rails 4 with Pundit & Statesman gem - 对象处于状态时的策略

AWS IAM 政策:仅标记未标记的资源

IAM 政策 - 如何引用资源?

政策不存在的 Laravel Gate 资源

资源和政策堆出来的联通还是衰落了,教训深刻