logo

pleroma

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

idempotency_plug.ex (2305B)


  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.Plugs.IdempotencyPlug do
  5. import Phoenix.Controller, only: [json: 2]
  6. import Plug.Conn
  7. @behaviour Plug
  8. @cachex Pleroma.Config.get([:cachex, :provider], Cachex)
  9. @impl true
  10. def init(opts), do: opts
  11. # Sending idempotency keys in `GET` and `DELETE` requests has no effect
  12. # and should be avoided, as these requests are idempotent by definition.
  13. @impl true
  14. def call(%{method: method} = conn, _) when method in ["POST", "PUT", "PATCH"] do
  15. case get_req_header(conn, "idempotency-key") do
  16. [key] -> process_request(conn, key)
  17. _ -> conn
  18. end
  19. end
  20. def call(conn, _), do: conn
  21. def process_request(conn, key) do
  22. case @cachex.get(:idempotency_cache, key) do
  23. {:ok, nil} ->
  24. cache_resposnse(conn, key)
  25. {:ok, record} ->
  26. send_cached(conn, key, record)
  27. {atom, message} when atom in [:ignore, :error] ->
  28. render_error(conn, message)
  29. end
  30. end
  31. defp cache_resposnse(conn, key) do
  32. register_before_send(conn, fn conn ->
  33. [request_id] = get_resp_header(conn, "x-request-id")
  34. content_type = get_content_type(conn)
  35. record = {request_id, content_type, conn.status, conn.resp_body}
  36. {:ok, _} = @cachex.put(:idempotency_cache, key, record)
  37. conn
  38. |> put_resp_header("idempotency-key", key)
  39. |> put_resp_header("x-original-request-id", request_id)
  40. end)
  41. end
  42. defp send_cached(conn, key, record) do
  43. {request_id, content_type, status, body} = record
  44. conn
  45. |> put_resp_header("idempotency-key", key)
  46. |> put_resp_header("idempotent-replayed", "true")
  47. |> put_resp_header("x-original-request-id", request_id)
  48. |> put_resp_content_type(content_type)
  49. |> send_resp(status, body)
  50. |> halt()
  51. end
  52. defp render_error(conn, message) do
  53. conn
  54. |> put_status(:unprocessable_entity)
  55. |> json(%{error: message})
  56. |> halt()
  57. end
  58. defp get_content_type(conn) do
  59. [content_type] = get_resp_header(conn, "content-type")
  60. if String.contains?(content_type, ";") do
  61. content_type
  62. |> String.split(";")
  63. |> hd()
  64. else
  65. content_type
  66. end
  67. end
  68. end