code-corps/code-corps-api

View on GitHub
lib/code_corps/policy/conversation.ex

Summary

Maintainability
Test Coverage
defmodule CodeCorps.Policy.Conversation do
  @moduledoc ~S"""
  Handles `CodeCorps.User` authorization of actions on `CodeCorps.Conversation`
  records.
  """

  import CodeCorps.Policy.Helpers,
    only: [administered_by?: 2, get_message: 1, get_project: 1]
  import Ecto.Query

  alias CodeCorps.{Conversation, Message, Project, ProjectUser, Repo, User}

  @spec scope(Ecto.Queryable.t, User.t) :: Ecto.Queryable.t
  def scope(queryable, %User{admin: true}), do: queryable
  def scope(queryable, %User{id: id}) do
    projects_administered_by_user_ids =
      Project
      |> join(:inner, [p], pu in ProjectUser, pu.project_id == p.id)
      |> where([_p, pu], pu.user_id == ^id)
      |> where([_p, pu], pu.role in ~w(admin owner))
      |> select([p], p.id)
      |> Repo.all

    scoped_message_ids =
      Message
      |> where([m], m.author_id == ^id)
      |> or_where([m], m.project_id in ^projects_administered_by_user_ids)
      |> select([m], m.id)
      |> Repo.all

    queryable
    |> where(user_id: ^id)
    |> or_where([c], c.message_id in ^scoped_message_ids)
  end

  def show?(%User{id: user_id}, %Conversation{user_id: target_user_id})
    when user_id == target_user_id do
    true
  end
  def show?(%User{} = user, %Conversation{} = conversation) do
    conversation |> get_message() |> get_project() |> administered_by?(user)
  end
  def show?(_, _), do: false

  def update?(%User{admin: true}, _conversation), do: true
  def update?(%User{} = user, %Conversation{} = conversation) do
    conversation |> get_message() |> get_project() |> administered_by?(user)
  end
  def update?(_, _), do: false
end