Commit aad3e7ac authored by Kamil Trzciński's avatar Kamil Trzciński

Merge branch '300661-fix-readiness-for-puma-single-2' into 'master'

Fix /-/readiness probe for Puma Single

See merge request gitlab-org/gitlab!53708
parents d2ca9373 3683ce9f
---
title: Fix /-/readiness probe for Puma Single
merge_request: 53708
author:
type: other
...@@ -4,7 +4,7 @@ def max_puma_workers ...@@ -4,7 +4,7 @@ def max_puma_workers
Puma.cli_config.options[:workers].to_i Puma.cli_config.options[:workers].to_i
end end
if Gitlab::Runtime.puma? && max_puma_workers == 0 if Gitlab::Runtime.puma? && !Gitlab::Runtime.puma_in_clustered_mode?
raise 'Puma is only supported in Clustered mode (workers > 0)' if Gitlab.com? raise 'Puma is only supported in Clustered mode (workers > 0)' if Gitlab.com?
warn 'WARNING: Puma is running in Single mode (workers = 0). Some features may not work. Please refer to https://gitlab.com/groups/gitlab-org/-/epics/5303 for info.' warn 'WARNING: Puma is running in Single mode (workers = 0). Some features may not work. Please refer to https://gitlab.com/groups/gitlab-org/-/epics/5303 for info.'
......
...@@ -11,6 +11,10 @@ module Gitlab ...@@ -11,6 +11,10 @@ module Gitlab
name.sub(/_check$/, '').capitalize name.sub(/_check$/, '').capitalize
end end
def available?
true
end
def readiness def readiness
raise NotImplementedError raise NotImplementedError
end end
......
...@@ -8,7 +8,16 @@ module Gitlab ...@@ -8,7 +8,16 @@ module Gitlab
extend SimpleAbstractCheck extend SimpleAbstractCheck
class << self class << self
extend ::Gitlab::Utils::Override
override :available?
def available?
Gitlab::Runtime.puma_in_clustered_mode?
end
def register_master def register_master
return unless available?
# when we fork, we pass the read pipe to child # when we fork, we pass the read pipe to child
# child can then react on whether the other end # child can then react on whether the other end
# of pipe is still available # of pipe is still available
...@@ -16,11 +25,15 @@ module Gitlab ...@@ -16,11 +25,15 @@ module Gitlab
end end
def finish_master def finish_master
return unless available?
close_read close_read
close_write close_write
end end
def register_worker def register_worker
return unless available?
# fork needs to close the pipe # fork needs to close the pipe
close_write close_write
end end
......
...@@ -48,6 +48,7 @@ module Gitlab ...@@ -48,6 +48,7 @@ module Gitlab
def probe_readiness def probe_readiness
checks checks
.select(&:available?)
.flat_map(&:readiness) .flat_map(&:readiness)
.compact .compact
.group_by(&:name) .group_by(&:name)
......
...@@ -12,6 +12,10 @@ module Gitlab ...@@ -12,6 +12,10 @@ module Gitlab
Gitlab::HealthChecks::Result.new( Gitlab::HealthChecks::Result.new(
'web_exporter', exporter.running) 'web_exporter', exporter.running)
end end
def available?
true
end
end end
attr_reader :running attr_reader :running
......
...@@ -81,6 +81,10 @@ module Gitlab ...@@ -81,6 +81,10 @@ module Gitlab
puma? || sidekiq? || action_cable? puma? || sidekiq? || action_cable?
end end
def puma_in_clustered_mode?
puma? && Puma.cli_config.options[:workers].to_i > 0
end
def max_threads def max_threads
threads = 1 # main thread threads = 1 # main thread
......
...@@ -4,11 +4,15 @@ require 'spec_helper' ...@@ -4,11 +4,15 @@ require 'spec_helper'
require_relative './simple_check_shared' require_relative './simple_check_shared'
RSpec.describe Gitlab::HealthChecks::MasterCheck do RSpec.describe Gitlab::HealthChecks::MasterCheck do
let(:result_class) { Gitlab::HealthChecks::Result }
before do before do
stub_const('SUCCESS_CODE', 100) stub_const('SUCCESS_CODE', 100)
stub_const('FAILURE_CODE', 101) stub_const('FAILURE_CODE', 101)
end
context 'when Puma runs in Clustered mode' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(true)
described_class.register_master described_class.register_master
end end
...@@ -16,7 +20,11 @@ RSpec.describe Gitlab::HealthChecks::MasterCheck do ...@@ -16,7 +20,11 @@ RSpec.describe Gitlab::HealthChecks::MasterCheck do
described_class.finish_master described_class.finish_master
end end
describe '#readiness' do describe '.available?' do
specify { expect(described_class.available?).to be true }
end
describe '.readiness' do
context 'when master is running' do context 'when master is running' do
it 'worker does return success' do it 'worker does return success' do
_, child_status = run_worker _, child_status = run_worker
...@@ -47,4 +55,16 @@ RSpec.describe Gitlab::HealthChecks::MasterCheck do ...@@ -47,4 +55,16 @@ RSpec.describe Gitlab::HealthChecks::MasterCheck do
Process.wait2(pid) Process.wait2(pid)
end end
end end
end
# '.readiness' check is not invoked if '.available?' returns false
context 'when Puma runs in Single mode' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(false)
end
describe '.available?' do
specify { expect(described_class.available?).to be false }
end
end
end end
...@@ -61,6 +61,35 @@ RSpec.describe Gitlab::HealthChecks::Probes::Collection do ...@@ -61,6 +61,35 @@ RSpec.describe Gitlab::HealthChecks::Probes::Collection do
expect(subject.json[:message]).to eq('Redis::CannotConnectError : Redis down') expect(subject.json[:message]).to eq('Redis::CannotConnectError : Redis down')
end end
end end
context 'when some checks are not available' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(false)
end
let(:checks) do
[
Gitlab::HealthChecks::MasterCheck
]
end
it 'asks for check availability' do
expect(Gitlab::HealthChecks::MasterCheck).to receive(:available?)
subject
end
it 'does not call `readiness` on checks that are not available' do
expect(Gitlab::HealthChecks::MasterCheck).not_to receive(:readiness)
subject
end
it 'does not fail collection check' do
expect(subject.http_status).to eq(200)
expect(subject.json[:status]).to eq('ok')
end
end
end end
context 'without checks' do context 'without checks' do
......
...@@ -44,10 +44,11 @@ RSpec.describe Gitlab::Runtime do ...@@ -44,10 +44,11 @@ RSpec.describe Gitlab::Runtime do
context "puma" do context "puma" do
let(:puma_type) { double('::Puma') } let(:puma_type) { double('::Puma') }
let(:max_workers) { 2 }
before do before do
stub_const('::Puma', puma_type) stub_const('::Puma', puma_type)
allow(puma_type).to receive_message_chain(:cli_config, :options).and_return(max_threads: 2) allow(puma_type).to receive_message_chain(:cli_config, :options).and_return(max_threads: 2, workers: max_workers)
stub_env('ACTION_CABLE_IN_APP', 'false') stub_env('ACTION_CABLE_IN_APP', 'false')
end end
...@@ -70,6 +71,20 @@ RSpec.describe Gitlab::Runtime do ...@@ -70,6 +71,20 @@ RSpec.describe Gitlab::Runtime do
it_behaves_like "valid runtime", :puma, 11 it_behaves_like "valid runtime", :puma, 11
end end
describe ".puma_in_clustered_mode?" do
context 'when Puma is set up with workers > 0' do
let(:max_workers) { 4 }
specify { expect(described_class.puma_in_clustered_mode?).to be true }
end
context 'when Puma is set up with workers = 0' do
let(:max_workers) { 0 }
specify { expect(described_class.puma_in_clustered_mode?).to be false }
end
end
end end
context "unicorn" do context "unicorn" do
......
...@@ -77,6 +77,11 @@ RSpec.describe HealthController do ...@@ -77,6 +77,11 @@ RSpec.describe HealthController do
shared_context 'endpoint responding with readiness data' do shared_context 'endpoint responding with readiness data' do
context 'when requesting instance-checks' do context 'when requesting instance-checks' do
context 'when Puma runs in Clustered mode' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(true)
end
it 'responds with readiness checks data' do it 'responds with readiness checks data' do
expect(Gitlab::HealthChecks::MasterCheck).to receive(:check) { true } expect(Gitlab::HealthChecks::MasterCheck).to receive(:check) { true }
...@@ -100,7 +105,23 @@ RSpec.describe HealthController do ...@@ -100,7 +105,23 @@ RSpec.describe HealthController do
end end
end end
context 'when Puma runs in Single mode' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(false)
end
it 'does not invoke MasterCheck, succeedes' do
expect(Gitlab::HealthChecks::MasterCheck).not_to receive(:check) { true }
subject
expect(json_response).to eq('status' => 'ok')
end
end
end
context 'when requesting all checks' do context 'when requesting all checks' do
shared_context 'endpoint responding with readiness data for all checks' do
before do before do
params.merge!(all: true) params.merge!(all: true)
end end
...@@ -164,6 +185,23 @@ RSpec.describe HealthController do ...@@ -164,6 +185,23 @@ RSpec.describe HealthController do
end end
end end
end end
context 'when Puma runs in Clustered mode' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(true)
end
it_behaves_like 'endpoint responding with readiness data for all checks'
end
context 'when Puma runs in Single mode' do
before do
allow(Gitlab::Runtime).to receive(:puma_in_clustered_mode?).and_return(false)
end
it_behaves_like 'endpoint responding with readiness data for all checks'
end
end
end end
context 'accessed from whitelisted ip' do context 'accessed from whitelisted ip' do
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment