grep -rl '# Copyright © .* Pleroma' * | xargs sed -i 's;Copyright © .* Pleroma .*;Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>;'
		
			
				
	
	
		
			86 lines
		
	
	
	
		
			2.3 KiB
		
	
	
	
		
			Elixir
		
	
	
	
	
	
			
		
		
	
	
			86 lines
		
	
	
	
		
			2.3 KiB
		
	
	
	
		
			Elixir
		
	
	
	
	
	
# Pleroma: A lightweight social networking server
 | 
						|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
 | 
						|
# SPDX-License-Identifier: AGPL-3.0-only
 | 
						|
 | 
						|
defmodule Pleroma.Web.Plugs.IdempotencyPlug do
 | 
						|
  import Phoenix.Controller, only: [json: 2]
 | 
						|
  import Plug.Conn
 | 
						|
 | 
						|
  @behaviour Plug
 | 
						|
 | 
						|
  @cachex Pleroma.Config.get([:cachex, :provider], Cachex)
 | 
						|
 | 
						|
  @impl true
 | 
						|
  def init(opts), do: opts
 | 
						|
 | 
						|
  # Sending idempotency keys in `GET` and `DELETE` requests has no effect
 | 
						|
  # and should be avoided, as these requests are idempotent by definition.
 | 
						|
 | 
						|
  @impl true
 | 
						|
  def call(%{method: method} = conn, _) when method in ["POST", "PUT", "PATCH"] do
 | 
						|
    case get_req_header(conn, "idempotency-key") do
 | 
						|
      [key] -> process_request(conn, key)
 | 
						|
      _ -> conn
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def call(conn, _), do: conn
 | 
						|
 | 
						|
  def process_request(conn, key) do
 | 
						|
    case @cachex.get(:idempotency_cache, key) do
 | 
						|
      {:ok, nil} ->
 | 
						|
        cache_resposnse(conn, key)
 | 
						|
 | 
						|
      {:ok, record} ->
 | 
						|
        send_cached(conn, key, record)
 | 
						|
 | 
						|
      {atom, message} when atom in [:ignore, :error] ->
 | 
						|
        render_error(conn, message)
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  defp cache_resposnse(conn, key) do
 | 
						|
    register_before_send(conn, fn conn ->
 | 
						|
      [request_id] = get_resp_header(conn, "x-request-id")
 | 
						|
      content_type = get_content_type(conn)
 | 
						|
 | 
						|
      record = {request_id, content_type, conn.status, conn.resp_body}
 | 
						|
      {:ok, _} = @cachex.put(:idempotency_cache, key, record)
 | 
						|
 | 
						|
      conn
 | 
						|
      |> put_resp_header("idempotency-key", key)
 | 
						|
      |> put_resp_header("x-original-request-id", request_id)
 | 
						|
    end)
 | 
						|
  end
 | 
						|
 | 
						|
  defp send_cached(conn, key, record) do
 | 
						|
    {request_id, content_type, status, body} = record
 | 
						|
 | 
						|
    conn
 | 
						|
    |> put_resp_header("idempotency-key", key)
 | 
						|
    |> put_resp_header("idempotent-replayed", "true")
 | 
						|
    |> put_resp_header("x-original-request-id", request_id)
 | 
						|
    |> put_resp_content_type(content_type)
 | 
						|
    |> send_resp(status, body)
 | 
						|
    |> halt()
 | 
						|
  end
 | 
						|
 | 
						|
  defp render_error(conn, message) do
 | 
						|
    conn
 | 
						|
    |> put_status(:unprocessable_entity)
 | 
						|
    |> json(%{error: message})
 | 
						|
    |> halt()
 | 
						|
  end
 | 
						|
 | 
						|
  defp get_content_type(conn) do
 | 
						|
    [content_type] = get_resp_header(conn, "content-type")
 | 
						|
 | 
						|
    if String.contains?(content_type, ";") do
 | 
						|
      content_type
 | 
						|
      |> String.split(";")
 | 
						|
      |> hd()
 | 
						|
    else
 | 
						|
      content_type
 | 
						|
    end
 | 
						|
  end
 | 
						|
end
 |