debian-mirror-gitlab/lib/gitlab/memory/watchdog/configurator.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

104 lines
4.2 KiB
Ruby
Raw Normal View History

2023-01-13 00:05:48 +05:30
# frozen_string_literal: true
module Gitlab
module Memory
class Watchdog
class Configurator
2023-03-04 22:38:38 +05:30
DEFAULT_PUMA_WORKER_RSS_LIMIT_MB = 1200
DEFAULT_SLEEP_INTERVAL_S = 60
DEFAULT_SIDEKIQ_SLEEP_INTERVAL_S = 3
MIN_SIDEKIQ_SLEEP_INTERVAL_S = 2
DEFAULT_MAX_STRIKES = 5
DEFAULT_MAX_HEAP_FRAG = 0.5
DEFAULT_MAX_MEM_GROWTH = 3.0
# grace_time / sleep_interval = max_strikes allowed for Sidekiq process to violate defined limits.
2023-04-23 21:23:45 +05:30
DEFAULT_SIDEKIQ_GRACE_TIME_S = 900
2023-03-04 22:38:38 +05:30
2023-01-13 00:05:48 +05:30
class << self
def configure_for_puma
2023-03-04 22:38:38 +05:30
->(config) do
2023-04-23 21:23:45 +05:30
config.handler = Gitlab::Memory::Watchdog::Handlers::PumaHandler.new
2023-03-04 22:38:38 +05:30
config.sleep_time_seconds = ENV.fetch('GITLAB_MEMWD_SLEEP_TIME_SEC', DEFAULT_SLEEP_INTERVAL_S).to_i
2023-01-13 00:05:48 +05:30
config.monitors(&configure_monitors_for_puma)
end
end
def configure_for_sidekiq
2023-03-04 22:38:38 +05:30
->(config) do
2023-04-23 21:23:45 +05:30
# Give Sidekiq up to 30 seconds to allow existing jobs to finish after exceeding the limit
shutdown_timeout_seconds = ENV.fetch('SIDEKIQ_MEMORY_KILLER_SHUTDOWN_WAIT', 30).to_i
config.handler = Gitlab::Memory::Watchdog::Handlers::SidekiqHandler.new(
shutdown_timeout_seconds,
sidekiq_sleep_time
)
2023-03-04 22:38:38 +05:30
config.sleep_time_seconds = sidekiq_sleep_time
2023-01-13 00:05:48 +05:30
config.monitors(&configure_monitors_for_sidekiq)
2023-03-04 22:38:38 +05:30
config.event_reporter = SidekiqEventReporter.new
2023-01-13 00:05:48 +05:30
end
end
private
def configure_monitors_for_puma
2023-03-04 22:38:38 +05:30
->(stack) do
max_strikes = ENV.fetch('GITLAB_MEMWD_MAX_STRIKES', DEFAULT_MAX_STRIKES).to_i
2023-01-13 00:05:48 +05:30
if Gitlab::Utils.to_boolean(ENV['DISABLE_PUMA_WORKER_KILLER'])
2023-03-04 22:38:38 +05:30
max_heap_frag = ENV.fetch('GITLAB_MEMWD_MAX_HEAP_FRAG', DEFAULT_MAX_HEAP_FRAG).to_f
max_mem_growth = ENV.fetch('GITLAB_MEMWD_MAX_MEM_GROWTH', DEFAULT_MAX_MEM_GROWTH).to_f
2023-01-13 00:05:48 +05:30
# stack.push MonitorClass, args*, max_strikes:, kwargs**, &block
stack.push Gitlab::Memory::Watchdog::Monitor::HeapFragmentation,
max_heap_fragmentation: max_heap_frag,
max_strikes: max_strikes
stack.push Gitlab::Memory::Watchdog::Monitor::UniqueMemoryGrowth,
max_mem_growth: max_mem_growth,
max_strikes: max_strikes
else
2023-03-04 22:38:38 +05:30
memory_limit = ENV.fetch('PUMA_WORKER_MAX_MEMORY', DEFAULT_PUMA_WORKER_RSS_LIMIT_MB).to_i
2023-01-13 00:05:48 +05:30
stack.push Gitlab::Memory::Watchdog::Monitor::RssMemoryLimit,
2023-03-04 22:38:38 +05:30
memory_limit_bytes: memory_limit.megabytes,
2023-01-13 00:05:48 +05:30
max_strikes: max_strikes
end
end
end
2023-03-04 22:38:38 +05:30
def sidekiq_sleep_time
[
ENV.fetch('SIDEKIQ_MEMORY_KILLER_CHECK_INTERVAL', DEFAULT_SIDEKIQ_SLEEP_INTERVAL_S).to_i,
MIN_SIDEKIQ_SLEEP_INTERVAL_S
].max
end
2023-01-13 00:05:48 +05:30
def configure_monitors_for_sidekiq
2023-03-04 22:38:38 +05:30
->(stack) do
if ENV['SIDEKIQ_MEMORY_KILLER_MAX_RSS'].to_i.nonzero?
soft_limit_bytes = ENV['SIDEKIQ_MEMORY_KILLER_MAX_RSS'].to_i.kilobytes
grace_time = ENV.fetch('SIDEKIQ_MEMORY_KILLER_GRACE_TIME', DEFAULT_SIDEKIQ_GRACE_TIME_S).to_i
max_strikes = grace_time / sidekiq_sleep_time
stack.push Gitlab::Memory::Watchdog::Monitor::RssMemoryLimit,
memory_limit_bytes: soft_limit_bytes,
max_strikes: max_strikes.to_i,
monitor_name: :rss_memory_soft_limit
end
if ENV['SIDEKIQ_MEMORY_KILLER_HARD_LIMIT_RSS'].to_i.nonzero?
hard_limit_bytes = ENV['SIDEKIQ_MEMORY_KILLER_HARD_LIMIT_RSS'].to_i.kilobytes
stack.push Gitlab::Memory::Watchdog::Monitor::RssMemoryLimit,
memory_limit_bytes: hard_limit_bytes,
max_strikes: 0,
monitor_name: :rss_memory_hard_limit
end
end
2023-01-13 00:05:48 +05:30
end
end
end
end
end
end