2023-12-15 17:29:02 +00:00
|
|
|
defmodule Pleroma.HTTP.BackoffTest do
|
|
|
|
@backoff_cache :http_backoff_cache
|
|
|
|
use Pleroma.DataCase, async: false
|
|
|
|
alias Pleroma.HTTP.Backoff
|
|
|
|
|
2024-05-06 23:34:48 +01:00
|
|
|
defp within_tolerance?(ttl, expected) do
|
2024-05-27 02:09:48 +01:00
|
|
|
ttl > expected - 15 and ttl < expected + 15
|
2024-05-06 23:34:48 +01:00
|
|
|
end
|
|
|
|
|
2023-12-15 17:29:02 +00:00
|
|
|
describe "get/3" do
|
|
|
|
test "should return {:ok, env} when not rate limited" do
|
|
|
|
Tesla.Mock.mock_global(fn
|
|
|
|
%Tesla.Env{url: "https://akkoma.dev/api/v1/instance"} ->
|
|
|
|
{:ok, %Tesla.Env{status: 200, body: "ok"}}
|
|
|
|
end)
|
2023-12-20 16:45:35 +00:00
|
|
|
|
2023-12-15 17:29:02 +00:00
|
|
|
assert {:ok, env} = Backoff.get("https://akkoma.dev/api/v1/instance")
|
|
|
|
assert env.status == 200
|
|
|
|
end
|
|
|
|
|
|
|
|
test "should return {:error, env} when rate limited" do
|
|
|
|
# Shove a value into the cache to simulate a rate limit
|
|
|
|
Cachex.put(@backoff_cache, "akkoma.dev", true)
|
2024-04-26 19:05:12 +01:00
|
|
|
assert {:error, :ratelimit} = Backoff.get("https://akkoma.dev/api/v1/instance")
|
2023-12-15 17:29:02 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
test "should insert a value into the cache when rate limited" do
|
|
|
|
Tesla.Mock.mock_global(fn
|
|
|
|
%Tesla.Env{url: "https://ratelimited.dev/api/v1/instance"} ->
|
|
|
|
{:ok, %Tesla.Env{status: 429, body: "Rate limited"}}
|
|
|
|
end)
|
|
|
|
|
2024-04-26 19:05:12 +01:00
|
|
|
assert {:error, :ratelimit} = Backoff.get("https://ratelimited.dev/api/v1/instance")
|
2023-12-15 17:29:02 +00:00
|
|
|
assert {:ok, true} = Cachex.get(@backoff_cache, "ratelimited.dev")
|
2023-12-20 16:45:35 +00:00
|
|
|
end
|
|
|
|
|
2024-05-06 23:36:00 +01:00
|
|
|
test "should insert a value into the cache when rate limited with a 503 response" do
|
|
|
|
Tesla.Mock.mock_global(fn
|
|
|
|
%Tesla.Env{url: "https://ratelimited.dev/api/v1/instance"} ->
|
|
|
|
{:ok, %Tesla.Env{status: 503, body: "Rate limited"}}
|
|
|
|
end)
|
|
|
|
|
|
|
|
assert {:error, :ratelimit} = Backoff.get("https://ratelimited.dev/api/v1/instance")
|
|
|
|
assert {:ok, true} = Cachex.get(@backoff_cache, "ratelimited.dev")
|
|
|
|
end
|
|
|
|
|
2023-12-20 16:45:35 +00:00
|
|
|
test "should parse the value of x-ratelimit-reset, if present" do
|
|
|
|
ten_minutes_from_now =
|
|
|
|
DateTime.utc_now() |> Timex.shift(minutes: 10) |> DateTime.to_iso8601()
|
|
|
|
|
|
|
|
Tesla.Mock.mock_global(fn
|
|
|
|
%Tesla.Env{url: "https://ratelimited.dev/api/v1/instance"} ->
|
|
|
|
{:ok,
|
|
|
|
%Tesla.Env{
|
|
|
|
status: 429,
|
|
|
|
body: "Rate limited",
|
|
|
|
headers: [{"x-ratelimit-reset", ten_minutes_from_now}]
|
|
|
|
}}
|
|
|
|
end)
|
|
|
|
|
2024-04-26 19:05:12 +01:00
|
|
|
assert {:error, :ratelimit} = Backoff.get("https://ratelimited.dev/api/v1/instance")
|
2023-12-20 16:45:35 +00:00
|
|
|
assert {:ok, true} = Cachex.get(@backoff_cache, "ratelimited.dev")
|
2024-05-06 23:34:48 +01:00
|
|
|
{:ok, ttl} = Cachex.ttl(@backoff_cache, "ratelimited.dev")
|
|
|
|
assert within_tolerance?(ttl, 600)
|
|
|
|
end
|
|
|
|
|
|
|
|
test "should parse the value of retry-after when it's a timestamp" do
|
|
|
|
ten_minutes_from_now =
|
|
|
|
DateTime.utc_now() |> Timex.shift(minutes: 10) |> DateTime.to_iso8601()
|
|
|
|
|
|
|
|
Tesla.Mock.mock_global(fn
|
|
|
|
%Tesla.Env{url: "https://ratelimited.dev/api/v1/instance"} ->
|
|
|
|
{:ok,
|
|
|
|
%Tesla.Env{
|
|
|
|
status: 429,
|
|
|
|
body: "Rate limited",
|
|
|
|
headers: [{"retry-after", ten_minutes_from_now}]
|
|
|
|
}}
|
|
|
|
end)
|
|
|
|
|
|
|
|
assert {:error, :ratelimit} = Backoff.get("https://ratelimited.dev/api/v1/instance")
|
|
|
|
assert {:ok, true} = Cachex.get(@backoff_cache, "ratelimited.dev")
|
|
|
|
{:ok, ttl} = Cachex.ttl(@backoff_cache, "ratelimited.dev")
|
|
|
|
assert within_tolerance?(ttl, 600)
|
|
|
|
end
|
|
|
|
|
|
|
|
test "should parse the value of retry-after when it's a number of seconds" do
|
|
|
|
Tesla.Mock.mock_global(fn
|
|
|
|
%Tesla.Env{url: "https://ratelimited.dev/api/v1/instance"} ->
|
|
|
|
{:ok,
|
|
|
|
%Tesla.Env{
|
|
|
|
status: 429,
|
|
|
|
body: "Rate limited",
|
|
|
|
headers: [{"retry-after", "600"}]
|
|
|
|
}}
|
|
|
|
end)
|
|
|
|
|
|
|
|
assert {:error, :ratelimit} = Backoff.get("https://ratelimited.dev/api/v1/instance")
|
|
|
|
assert {:ok, true} = Cachex.get(@backoff_cache, "ratelimited.dev")
|
|
|
|
# assert that the value is 10 minutes from now
|
|
|
|
{:ok, ttl} = Cachex.ttl(@backoff_cache, "ratelimited.dev")
|
|
|
|
assert within_tolerance?(ttl, 600)
|
2023-12-20 16:45:35 +00:00
|
|
|
end
|
2023-12-15 17:29:02 +00:00
|
|
|
end
|
|
|
|
end
|