Skip to content

Commit

Permalink
Add assign_as option for Plug.RequestId
Browse files Browse the repository at this point in the history
- `:assign_as` - The name of the key that will be used to store the
  discovered or generated request id in `conn.private`. If not provided,
  the request id will not be stored.

  ```elixir
  plug Plug.RequestId, assign_as: :plug_request_id
  ```

Resolves: #1171
  • Loading branch information
halostatue committed Oct 6, 2023
1 parent 44747cd commit cefbb65
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 7 deletions.
24 changes: 17 additions & 7 deletions lib/plug/request_id.ex
Original file line number Diff line number Diff line change
Expand Up @@ -34,22 +34,29 @@ defmodule Plug.RequestId do
plug Plug.RequestId, http_header: "custom-request-id"
* `:assign_as` - The name of the key that will be used to store the
discovered or generated request id in `conn.private`. If not provided,
the request id will not be stored.
plug Plug.RequestId, assign_as: :plug_request_id
"""

require Logger
alias Plug.Conn
@behaviour Plug

@impl true
def init(opts) do
Keyword.get(opts, :http_header, "x-request-id")
end
def init(opts), do: opts

@impl true
def call(conn, req_id_header) do
def call(conn, opts) do
header = Keyword.get(opts, :http_header, "x-request-id")
assign_as = Keyword.get(opts, :assign_as)

conn
|> get_request_id(req_id_header)
|> set_request_id(req_id_header)
|> get_request_id(header)
|> set_request_id(header, assign_as)
end

defp get_request_id(conn, header) do
Expand All @@ -59,8 +66,11 @@ defmodule Plug.RequestId do
end
end

defp set_request_id({conn, request_id}, header) do
defp set_request_id({conn, request_id}, header, assign_as) do
Logger.metadata(request_id: request_id)

conn = if assign_as, do: Conn.put_private(conn, assign_as, request_id), else: conn

Conn.put_resp_header(conn, header, request_id)
end

Expand Down
16 changes: 16 additions & 0 deletions test/plug/request_id_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,22 @@ defmodule Plug.RequestIdTest do
assert res_request_id == meta_request_id
end

test "assigns the request id to conn.private when given an assignment key" do
request_id = "existingidthatislongenough"

conn =
conn(:get, "/")
|> put_req_header("x-request-id", request_id)
|> call(assign_as: :plug_request_id)

[res_request_id] = get_resp_header(conn, "x-request-id")
meta_request_id = Logger.metadata()[:request_id]
assigned_request_id = conn.private.plug_request_id
assert assigned_request_id == request_id
assert res_request_id == assigned_request_id
assert res_request_id == meta_request_id
end

defp generated_request_id?(request_id) do
Regex.match?(~r/\A[A-Za-z0-9-_]+\z/, request_id)
end
Expand Down

0 comments on commit cefbb65

Please sign in to comment.