closing idle gun connections
This commit is contained in:
parent
07e7c80bc9
commit
96bc967008
@ -608,6 +608,8 @@ config :pleroma, Pleroma.Repo,
|
|||||||
config :pleroma, :connections_pool,
|
config :pleroma, :connections_pool,
|
||||||
checkin_timeout: 250,
|
checkin_timeout: 250,
|
||||||
max_connections: 250,
|
max_connections: 250,
|
||||||
|
max_idle_time: 10,
|
||||||
|
closing_idle_conns_interval: 10,
|
||||||
retry: 1,
|
retry: 1,
|
||||||
retry_timeout: 1000,
|
retry_timeout: 1000,
|
||||||
await_up_timeout: 5_000
|
await_up_timeout: 5_000
|
||||||
|
@ -394,15 +394,20 @@ For each pool, the options are:
|
|||||||
|
|
||||||
Advanced settings for connections pool. Pool with opened connections. These connections can be reused in worker pools.
|
Advanced settings for connections pool. Pool with opened connections. These connections can be reused in worker pools.
|
||||||
|
|
||||||
For big instances it's recommended to increase `config :pleroma, :connections_pool, max_connections: 500` up to 500-1000.
|
For big instances it's recommended to increase `max_connections` up to 500-1000. It will increase memory usage, but federation would work faster.
|
||||||
It will increase memory usage, but federation would work faster.
|
|
||||||
|
|
||||||
* `:checkin_timeout` - timeout to checkin connection from pool. Default: 250ms.
|
* `:checkin_timeout` - timeout to checkin connection from pool. Default: 250ms.
|
||||||
* `:max_connections` - maximum number of connections in the pool. Default: 250 connections.
|
* `:max_connections` - maximum number of connections in the pool. Default: 250 connections.
|
||||||
|
* `:max_idle_time` - maximum of time, while connection can be idle. Default: 10 minutes.
|
||||||
|
* `:closing_idle_conns_interval` - interval between cleaning pool from idle connections. Default: 10 minutes.
|
||||||
* `:retry` - number of retries, while `gun` will try to reconnect if connection goes down. Default: 1.
|
* `:retry` - number of retries, while `gun` will try to reconnect if connection goes down. Default: 1.
|
||||||
* `:retry_timeout` - time between retries when `gun` will try to reconnect in milliseconds. Default: 1000ms.
|
* `:retry_timeout` - time between retries when `gun` will try to reconnect in milliseconds. Default: 1000ms.
|
||||||
* `:await_up_timeout` - timeout while `gun` will wait until connection is up. Default: 5000ms.
|
* `:await_up_timeout` - timeout while `gun` will wait until connection is up. Default: 5000ms.
|
||||||
|
|
||||||
|
*If you are increasing `max_connections` setting, dont't forget to increase limit for file descriptors:*
|
||||||
|
* `installation/pleroma.service` - `LimitNOFILE`
|
||||||
|
* `installation/pleroma.supervisord` - `minfds`
|
||||||
|
|
||||||
### :pools
|
### :pools
|
||||||
|
|
||||||
*For `gun` adapter*
|
*For `gun` adapter*
|
||||||
|
@ -12,6 +12,7 @@ defmodule Pleroma.Pool.Connections do
|
|||||||
|
|
||||||
@type domain :: String.t()
|
@type domain :: String.t()
|
||||||
@type conn :: Pleroma.Gun.Conn.t()
|
@type conn :: Pleroma.Gun.Conn.t()
|
||||||
|
@type seconds :: pos_integer()
|
||||||
|
|
||||||
@type t :: %__MODULE__{
|
@type t :: %__MODULE__{
|
||||||
conns: %{domain() => conn()},
|
conns: %{domain() => conn()},
|
||||||
@ -26,7 +27,10 @@ defmodule Pleroma.Pool.Connections do
|
|||||||
end
|
end
|
||||||
|
|
||||||
@impl true
|
@impl true
|
||||||
def init(opts), do: {:ok, %__MODULE__{conns: %{}, opts: opts}}
|
def init(opts) do
|
||||||
|
schedule_close_idle_conns()
|
||||||
|
{:ok, %__MODULE__{conns: %{}, opts: opts}}
|
||||||
|
end
|
||||||
|
|
||||||
@spec checkin(String.t() | URI.t(), atom()) :: pid() | nil
|
@spec checkin(String.t() | URI.t(), atom()) :: pid() | nil
|
||||||
def checkin(url, name)
|
def checkin(url, name)
|
||||||
@ -154,16 +158,16 @@ defmodule Pleroma.Pool.Connections do
|
|||||||
def handle_call(:unused_conns, _from, state) do
|
def handle_call(:unused_conns, _from, state) do
|
||||||
unused_conns =
|
unused_conns =
|
||||||
state.conns
|
state.conns
|
||||||
|> Enum.filter(&filter_conns/1)
|
|> Enum.filter(&idle_conn?/1)
|
||||||
|> Enum.sort(&sort_conns/2)
|
|> Enum.sort(&least_used/2)
|
||||||
|
|
||||||
{:reply, unused_conns, state}
|
{:reply, unused_conns, state}
|
||||||
end
|
end
|
||||||
|
|
||||||
defp filter_conns({_, %{conn_state: :idle, used_by: []}}), do: true
|
defp idle_conn?({_, %{conn_state: :idle}}), do: true
|
||||||
defp filter_conns(_), do: false
|
defp idle_conn?(_), do: false
|
||||||
|
|
||||||
defp sort_conns({_, c1}, {_, c2}) do
|
defp least_used({_, c1}, {_, c2}) do
|
||||||
c1.crf <= c2.crf and c1.last_reference <= c2.last_reference
|
c1.crf <= c2.crf and c1.last_reference <= c2.last_reference
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -265,6 +269,38 @@ defmodule Pleroma.Pool.Connections do
|
|||||||
{:noreply, state}
|
{:noreply, state}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@impl true
|
||||||
|
def handle_info({:close_idle_conns, max_idle_time}, state) do
|
||||||
|
closing_time = :os.system_time(:second) - max_idle_time
|
||||||
|
|
||||||
|
idle_conns_keys =
|
||||||
|
state.conns
|
||||||
|
|> Enum.filter(&idle_more_than?(&1, closing_time))
|
||||||
|
|> Enum.map(fn {key, %{conn: conn}} ->
|
||||||
|
Gun.close(conn)
|
||||||
|
key
|
||||||
|
end)
|
||||||
|
|
||||||
|
schedule_close_idle_conns()
|
||||||
|
{:noreply, put_in(state.conns, Map.drop(state.conns, idle_conns_keys))}
|
||||||
|
end
|
||||||
|
|
||||||
|
defp schedule_close_idle_conns do
|
||||||
|
max_idle_time = Config.get([:connections_pool, :max_idle_time], 10) * 60
|
||||||
|
interval = Config.get([:connections_pool, :closing_idle_conns_interval], 10) * 60 * 1000
|
||||||
|
Process.send_after(self(), {:close_idle_conns, max_idle_time}, interval)
|
||||||
|
end
|
||||||
|
|
||||||
|
defp idle_more_than?(
|
||||||
|
{_, %{conn_state: :idle, last_reference: idle_since}},
|
||||||
|
closing_time
|
||||||
|
)
|
||||||
|
when closing_time >= idle_since do
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
|
defp idle_more_than?(_, _), do: false
|
||||||
|
|
||||||
defp find_conn(conns, conn_pid) do
|
defp find_conn(conns, conn_pid) do
|
||||||
Enum.find(conns, fn {_key, conn} ->
|
Enum.find(conns, fn {_key, conn} ->
|
||||||
conn.conn == conn_pid
|
conn.conn == conn_pid
|
||||||
|
@ -757,4 +757,38 @@ defmodule Pleroma.Pool.ConnectionsTest do
|
|||||||
Connections.remove_conn(name, "1")
|
Connections.remove_conn(name, "1")
|
||||||
assert Connections.count(name) == 0
|
assert Connections.count(name) == 0
|
||||||
end
|
end
|
||||||
|
|
||||||
|
test "close_idle_conns/2", %{name: name} do
|
||||||
|
GunMock
|
||||||
|
|> expect(:close, fn _ -> :ok end)
|
||||||
|
|> allow(self(), name)
|
||||||
|
|
||||||
|
Connections.add_conn(name, "1", %Conn{
|
||||||
|
conn_state: :idle,
|
||||||
|
last_reference: now() - 30,
|
||||||
|
conn: self()
|
||||||
|
})
|
||||||
|
|
||||||
|
Connections.add_conn(name, "2", %Conn{
|
||||||
|
conn_state: :idle,
|
||||||
|
last_reference: now() - 10,
|
||||||
|
conn: self()
|
||||||
|
})
|
||||||
|
|
||||||
|
Connections.add_conn(name, "3", %Conn{
|
||||||
|
conn_state: :active,
|
||||||
|
conn: self()
|
||||||
|
})
|
||||||
|
|
||||||
|
name
|
||||||
|
|> Process.whereis()
|
||||||
|
|> send({:close_idle_conns, 15})
|
||||||
|
|
||||||
|
assert %Connections{
|
||||||
|
conns: %{
|
||||||
|
"3" => %Conn{},
|
||||||
|
"2" => %Conn{}
|
||||||
|
}
|
||||||
|
} = Connections.get_state(name)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
Loading…
Reference in New Issue
Block a user