Skip to content

Commit

Permalink
Update Req.Utils
Browse files Browse the repository at this point in the history
  • Loading branch information
wojtekmach committed Nov 22, 2024
1 parent 2a80282 commit 5bfbccc
Show file tree
Hide file tree
Showing 5 changed files with 93 additions and 35 deletions.
4 changes: 2 additions & 2 deletions lib/req.ex
Original file line number Diff line number Diff line change
Expand Up @@ -1357,12 +1357,12 @@ defmodule Req do
end

defp encode_header_value(%DateTime{} = datetime) do
datetime |> DateTime.shift_zone!("Etc/UTC") |> Req.Utils.format_http_datetime()
datetime |> DateTime.shift_zone!("Etc/UTC") |> Req.Utils.format_http_date()
end

defp encode_header_value(%NaiveDateTime{} = datetime) do
IO.warn("setting header to %NaiveDateTime{} is deprecated, use %DateTime{} instead")
Req.Utils.format_http_datetime(datetime)
Req.Utils.format_http_date(datetime)
end

defp encode_header_value(value) when is_binary(value) do
Expand Down
30 changes: 1 addition & 29 deletions lib/req/response.ex
Original file line number Diff line number Diff line change
Expand Up @@ -269,37 +269,9 @@ defmodule Req.Response do

:error ->
delay_value
|> parse_http_datetime()
|> Req.Utils.parse_http_date!()
|> DateTime.diff(DateTime.utc_now(), :millisecond)
|> max(0)
end
end

@month_numbers %{
"Jan" => "01",
"Feb" => "02",
"Mar" => "03",
"Apr" => "04",
"May" => "05",
"Jun" => "06",
"Jul" => "07",
"Aug" => "08",
"Sep" => "09",
"Oct" => "10",
"Nov" => "11",
"Dec" => "12"
}

defp parse_http_datetime(datetime) do
[_day_of_week, day, month, year, time, "GMT"] = String.split(datetime, " ")
date = year <> "-" <> @month_numbers[month] <> "-" <> day

case DateTime.from_iso8601(date <> " " <> time <> "Z") do
{:ok, valid_datetime, 0} ->
valid_datetime

{:error, reason} ->
raise "cannot parse \"retry-after\" header value #{inspect(datetime)} as datetime, reason: #{reason}"
end
end
end
2 changes: 1 addition & 1 deletion lib/req/steps.ex
Original file line number Diff line number Diff line change
Expand Up @@ -628,7 +628,7 @@ defmodule Req.Steps do
stat.mtime
|> NaiveDateTime.from_erl!()
|> DateTime.from_naive!("Etc/UTC")
|> Req.Utils.format_http_datetime()
|> Req.Utils.format_http_date()

Req.Request.put_new_header(request, "if-modified-since", http_datetime_string)

Expand Down
90 changes: 88 additions & 2 deletions lib/req/utils.ex
Original file line number Diff line number Diff line change
Expand Up @@ -253,13 +253,99 @@ defmodule Req.Utils do
## Examples
iex> Req.Utils.format_http_datetime(~U[2024-01-01 09:00:00Z])
iex> Req.Utils.format_http_date(~U[2024-01-01 09:00:00Z])
"Mon, 01 Jan 2024 09:00:00 GMT"
"""
def format_http_datetime(datetime) do
def format_http_date(datetime) do
Calendar.strftime(datetime, "%a, %d %b %Y %H:%M:%S GMT")
end

@doc """
Parses "HTTP Date" as datetime.
## Examples
iex> Req.Utils.parse_http_date("Mon, 01 Jan 2024 09:00:00 GMT")
{:ok, ~U[2024-01-01 09:00:00Z]}
"""
def parse_http_date(<<
day_name::binary-size(3),
", ",
day::binary-size(2),
" ",
month_name::binary-size(3),
" ",
year::binary-size(4),
" ",
time::binary-size(8),
" GMT"
>>) do
with {:ok, day_of_week} <- parse_day_name(day_name),
{day, ""} <- Integer.parse(day),
{:ok, month} <- parse_month_name(month_name),
{year, ""} <- Integer.parse(year),
{:ok, time} <- Time.from_iso8601(time),
{:ok, date} <- Date.new(year, month, day),
true <- day_of_week == Date.day_of_week(date) do
DateTime.new(date, time)
else
{:error, _} = e ->
e

_ ->
{:error, :invalid_format}
end
end

def parse_http_date(binary) when is_binary(binary) do
{:error, :invalid_format}
end

defp parse_month_name("Jan"), do: {:ok, 1}
defp parse_month_name("Feb"), do: {:ok, 2}
defp parse_month_name("Mar"), do: {:ok, 3}
defp parse_month_name("Apr"), do: {:ok, 4}
defp parse_month_name("May"), do: {:ok, 5}
defp parse_month_name("Jun"), do: {:ok, 6}
defp parse_month_name("Jul"), do: {:ok, 7}
defp parse_month_name("Aug"), do: {:ok, 8}
defp parse_month_name("Sep"), do: {:ok, 9}
defp parse_month_name("Oct"), do: {:ok, 10}
defp parse_month_name("Nov"), do: {:ok, 11}
defp parse_month_name("Dec"), do: {:ok, 12}
defp parse_month_name(_), do: :error

defp parse_day_name("Mon"), do: {:ok, 1}
defp parse_day_name("Tue"), do: {:ok, 2}
defp parse_day_name("Wed"), do: {:ok, 3}
defp parse_day_name("Thu"), do: {:ok, 4}
defp parse_day_name("Fri"), do: {:ok, 5}
defp parse_day_name("Sat"), do: {:ok, 6}
defp parse_day_name("Sun"), do: {:ok, 7}
defp parse_day_name(_), do: :error

@doc """
Parses "HTTP Date" as datetime or raises an error.
## Examples
iex> Req.Utils.parse_http_date!("Mon, 01 Jan 2024 09:00:00 GMT")
~U[2024-01-01 09:00:00Z]
iex> Req.Utils.parse_http_date!("Mon")
** (ArgumentError) cannot parse "Mon" as HTTP date, reason: :invalid_format
"""
def parse_http_date!(binary) do
case parse_http_date(binary) do
{:ok, datetime} ->
datetime

{:error, reason} ->
raise ArgumentError,
"cannot parse #{inspect(binary)} as HTTP date, reason: #{inspect(reason)}"
end
end

@doc """
Returns a stream where each element is gzipped.
Expand Down
2 changes: 1 addition & 1 deletion test/req/steps_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -1549,7 +1549,7 @@ defmodule Req.StepsTest do

defp retry_after(r, value), do: Req.Response.put_header(r, "retry-after", retry_after(value))
defp retry_after(integer) when is_integer(integer), do: to_string(integer)
defp retry_after(%DateTime{} = dt), do: Req.Utils.format_http_datetime(dt)
defp retry_after(%DateTime{} = dt), do: Req.Utils.format_http_date(dt)

@tag :capture_log
test "always failing", c do
Expand Down

0 comments on commit 5bfbccc

Please sign in to comment.