lib/history_web/live/predicted_funding_rate_job/index_live.ex

defmodule HistoryWeb.PredictedFundingRateJob.IndexLive do
  use HistoryWeb, :live_view
  import HistoryWeb.JobView
  alias History.{PredictedFundingRateJobs, Products}
  alias History.PredictedFundingRates.PredictedFundingRateJob

  @impl true
  def mount(_params, _session, socket) do
    Phoenix.PubSub.subscribe(Tai.PubSub, "predicted_funding_rate_job:*")

    socket =
      socket
      |> assign_swap_products()
      |> assign(:job_changeset, PredictedFundingRateJobs.changeset(%{}))

    {:ok, socket}
  end

  @impl true
  def handle_params(params, _, socket) do
    {page, _} = params |> Map.get("page", "1") |> Integer.parse()
    {page_size, _} = params |> Map.get("page_size", "25") |> Integer.parse()

    socket =
      socket
      |> assign(page_size: page_size)
      |> assign(current_page: page)
      |> assign_latest()

    {:noreply, socket}
  end

  @impl true
  def handle_event("download", params, socket) do
    job_params = Map.get(params, "predicted_funding_rate_job", %{})

    products =
      job_params
      |> Map.get("products", [])
      |> Enum.map(&Jason.decode!/1)

    attrs =
      job_params
      |> Map.put("products", products)
      |> Map.put("status", "enqueued")

    socket =
      with {:ok, job} <- PredictedFundingRateJobs.insert(attrs) do
        socket
        |> assign(
          :job_changeset,
          PredictedFundingRateJob.changeset(job, %{})
        )
        |> assign_latest()
      else
        {:error, changeset} ->
          socket
          |> assign(:job_changeset, changeset)
      end

    {:noreply, socket}
  end

  @impl true
  def handle_info({:predicted_funding_rate_job, :update, _}, socket) do
    jobs =
      PredictedFundingRateJobs.latest(
        page: socket.assigns.current_page,
        page_size: socket.assigns.page_size
      )

    socket =
      socket
      |> assign(:jobs, jobs)

    {:noreply, socket}
  end

  defp assign_latest(socket) do
    first_page = 1
    current_page = socket.assigns.current_page
    previous_page = max(current_page - 1, first_page)
    last_page = ceil(PredictedFundingRateJobs.count() / socket.assigns.page_size)
    next_page = min(current_page + 1, last_page)

    socket
    |> assign(current_page: current_page)
    |> assign(first_page: first_page)
    |> assign(previous_page: previous_page)
    |> assign(last_page: last_page)
    |> assign(next_page: next_page)
    |> assign(
      :jobs,
      PredictedFundingRateJobs.latest(
        page: socket.assigns.current_page,
        page_size: socket.assigns.page_size
      )
    )
  end

  defp assign_swap_products(socket) do
    socket
    |> assign(:swap_products, Products.swap())
  end
end