lib/credo/cli/command/info/info_command.ex

defmodule Credo.CLI.Command.Info.InfoCommand do
  @moduledoc false

  use Credo.CLI.Command,
    short_description: "Show useful debug information",
    cli_switches: Credo.CLI.Command.Suggest.SuggestCommand.cli_switches()

  alias Credo.CLI.Command.Info.InfoOutput
  alias Credo.CLI.Task
  alias Credo.Execution

  def init(exec) do
    Execution.put_pipeline(exec, "info",
      load_and_validate_source_files: [
        {Task.LoadAndValidateSourceFiles, []}
      ],
      prepare_analysis: [
        {Task.PrepareChecksToRun, []}
      ],
      print_info: [
        {__MODULE__.PrintInfo, []}
      ]
    )
  end

  @doc false
  def call(%Execution{help: true} = exec, _opts), do: InfoOutput.print_help(exec)
  def call(exec, _opts), do: Execution.run_pipeline(exec, __MODULE__)

  defmodule PrintInfo do
    def call(exec, _opts \\ []) do
      InfoOutput.print(exec, info(exec))

      exec
    end

    defp info(exec) do
      %{
        "system" => %{
          "credo" => Credo.version(),
          "elixir" => System.version(),
          "erlang" => System.otp_release()
        },
        "config" => %{
          "plugins" => plugins(exec),
          "checks" => checks(exec),
          "files" => files(exec)
        }
      }
    end

    defp plugins(exec) do
      Enum.map(exec.plugins, fn {name, params} ->
        %{"name" => name, "params" => Enum.into(params, %{})}
      end)
    end

    defp checks(exec) do
      {checks, _only_matching, _ignore_matching} = Execution.checks(exec)

      Enum.map(checks, fn {name, params} ->
        %{"name" => name, "params" => Enum.into(params, %{})}
      end)
    end

    defp files(exec) do
      exec
      |> Execution.get_source_files()
      |> Enum.map(& &1.filename)
    end
  end
end