logo

pleroma

My custom branche(s) on git.pleroma.social/pleroma/pleroma git clone https://hacktivis.me/git/pleroma.git

poll_view.ex (3026B)


  1. # Pleroma: A lightweight social networking server
  2. # Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
  3. # SPDX-License-Identifier: AGPL-3.0-only
  4. defmodule Pleroma.Web.MastodonAPI.PollView do
  5. use Pleroma.Web, :view
  6. alias Pleroma.Web.CommonAPI.Utils
  7. def render("show.json", %{object: object, multiple: multiple, options: options} = params) do
  8. {end_time, expired} = end_time_and_expired(object)
  9. {options, votes_count} = options_and_votes_count(options)
  10. poll = %{
  11. # Mastodon uses separate ids for polls, but an object can't have
  12. # more than one poll embedded so object id is fine
  13. id: to_string(object.id),
  14. expires_at: end_time,
  15. expired: expired,
  16. multiple: multiple,
  17. votes_count: votes_count,
  18. voters_count: voters_count(object),
  19. options: options,
  20. emojis: Pleroma.Web.MastodonAPI.StatusView.build_emojis(object.data["emoji"])
  21. }
  22. if params[:for] do
  23. # when unauthenticated Mastodon doesn't include `voted` & `own_votes` keys in response
  24. {voted, own_votes} = voted_and_own_votes(params, options)
  25. Map.merge(poll, %{voted: voted, own_votes: own_votes})
  26. else
  27. poll
  28. end
  29. end
  30. def render("show.json", %{object: object} = params) do
  31. case object.data do
  32. %{"anyOf" => [_ | _] = options} ->
  33. render(__MODULE__, "show.json", Map.merge(params, %{multiple: true, options: options}))
  34. %{"oneOf" => [_ | _] = options} ->
  35. render(__MODULE__, "show.json", Map.merge(params, %{multiple: false, options: options}))
  36. _ ->
  37. nil
  38. end
  39. end
  40. defp end_time_and_expired(object) do
  41. if object.data["closed"] do
  42. end_time = NaiveDateTime.from_iso8601!(object.data["closed"])
  43. expired = NaiveDateTime.compare(end_time, NaiveDateTime.utc_now()) == :lt
  44. {Utils.to_masto_date(end_time), expired}
  45. else
  46. {nil, false}
  47. end
  48. end
  49. defp options_and_votes_count(options) do
  50. Enum.map_reduce(options, 0, fn %{"name" => name} = option, count ->
  51. current_count = option["replies"]["totalItems"] || 0
  52. {%{
  53. title: name,
  54. votes_count: current_count
  55. }, current_count + count}
  56. end)
  57. end
  58. defp voters_count(%{data: %{"voters" => [_ | _] = voters}}) do
  59. length(voters)
  60. end
  61. defp voters_count(_), do: 0
  62. defp voted_and_own_votes(%{object: object} = params, options) do
  63. if params[:for] do
  64. existing_votes =
  65. Pleroma.Web.ActivityPub.Utils.get_existing_votes(params[:for].ap_id, object)
  66. voted = existing_votes != [] or params[:for].ap_id == object.data["actor"]
  67. own_votes =
  68. if voted do
  69. titles = Enum.map(options, & &1[:title])
  70. Enum.reduce(existing_votes, [], fn vote, acc ->
  71. data = vote |> Map.get(:object) |> Map.get(:data)
  72. index = Enum.find_index(titles, &(&1 == data["name"]))
  73. [index | acc]
  74. end)
  75. else
  76. []
  77. end
  78. {voted, own_votes}
  79. else
  80. {false, []}
  81. end
  82. end
  83. end