code-corps/code-corps-api

View on GitHub
lib/code_corps/comment/service.ex

Summary

Maintainability
Test Coverage
defmodule CodeCorps.Comment.Service do
  @moduledoc ~S"""
  In charge of performing CRUD operations on `Comment` records, as well as any
  additional actions that need to be performed when such an operation happens.
  """

  alias CodeCorps.{
    Comment,
    GitHub,
    GitHub.Sync,
    GithubComment,
    GithubIssue,
    Task,
    Repo
  }
  alias Ecto.{Changeset, Multi}

  require Logger

  @type record_result ::
          {:ok, Comment.t} | {:error, Changeset.t} | {:error, GitHub.api_error_struct()}

  # :user, :github_issue and :github_repo are required for connecting to github
  # :project and :organization are required in order to add a header to the
  # github comment body when the user themselves are not connected to github,
  # but the parent task is
  #
  # Right now, all of these preloads are loaded at once. If there are
  # performance issues, we can split them up according the the information
  # provided here.
  @preloads [
    :github_comment,
    :user,
    task: [
      :github_issue,
      [github_repo: :github_app_installation],
      [project: :organization]
    ]
  ]

  @doc ~S"""
  Creates a `Comment` record using the provided parameters

  Also creates comment on GitHub if associated `Task` is github-connected.
  """
  @spec create(map) :: {:ok, Comment.t} | {:error, Changeset.t}
  def create(%{} = attributes) do
    Multi.new()
    |> Multi.insert(:comment, %Comment{} |> Comment.create_changeset(attributes))
    |> Multi.run(:preload, fn %{comment: %Comment{} = comment} ->
         {:ok, comment |> Repo.preload(@preloads)}
       end)
    |> Multi.run(:github, fn %{preload: %Comment{} = comment} -> comment |> create_on_github() end)
    |> Repo.transaction()
    |> marshall_result
  end

  @doc ~S"""
  Updates the provided `Comment` record using the provided parameters
  """
  @spec update(Comment.t, map) :: {:ok, Comment.t} | {:error, Changeset.t}
  def update(%Comment{} = comment, %{} = attributes) do
    Multi.new()
    |> Multi.update(:comment, comment |> Comment.update_changeset(attributes))
    |> Multi.run(:preload, fn %{comment: %Comment{} = comment} ->
         {:ok, comment |> Repo.preload(@preloads)}
       end)
    |> Multi.run(:github, fn %{preload: %Comment{} = comment} -> comment |> update_on_github() end)
    |> Repo.transaction()
    |> marshall_result()
  end

  @spec marshall_result(tuple) :: {:ok, Comment.t} | {:error, Changeset.t} | {:error, :github}
  defp marshall_result({:ok, %{github: %Comment{} = comment}}), do: {:ok, comment}
  defp marshall_result({:error, :comment, %Changeset{} = changeset, _steps}), do: {:error, changeset}
  defp marshall_result({:error, :github, result, _steps}) do
    Logger.info("An error occurred when creating/updating the comment with the GitHub API")
    Logger.info("#{inspect(result)}")
    {:error, :github}
  end

  @spec create_on_github(Comment.t) :: record_result
  defp create_on_github(%Comment{task: %Task{github_issue_id: nil}} = comment), do: {:ok, comment}
  defp create_on_github(%Comment{task: %Task{github_issue: github_issue}} = comment) do
    with {:ok, payload} <- comment |> GitHub.API.Comment.create(),
         {:ok, %GithubComment{} = github_comment} <-
           Sync.GithubComment.create_or_update_comment(github_issue, payload) do
      comment |> link_with_github_changeset(github_comment) |> Repo.update()
    else
      {:error, error} -> {:error, error}
    end
  end

  @spec link_with_github_changeset(Comment.t, GithubComment.t) :: Changeset.t
  defp link_with_github_changeset(%Comment{} = comment, %GithubComment{} = github_comment) do
    comment |> Changeset.change(%{github_comment: github_comment})
  end

  @spec update_on_github(Comment.t) :: record_result
  defp update_on_github(%Comment{github_comment_id: nil} = comment), do: {:ok, comment}
  defp update_on_github(
         %Comment{task: %Task{github_issue: %GithubIssue{} = github_issue}} = comment
       ) do
    with {:ok, payload} <- comment |> GitHub.API.Comment.update(),
         {:ok, %GithubComment{}} <-
           Sync.GithubComment.create_or_update_comment(github_issue, payload) do
      {:ok, comment}
    else
      {:error, error} -> {:error, error}
    end
  end
end