Skip to content

Commit

Permalink
MONGOID-5758: Add Mongoid.reconnect_clients and improve forking webse…
Browse files Browse the repository at this point in the history
…rver documentation (#5808)

* This PR does the following:
- Add Mongoid.reconnect_clients (analogous to Mongoid.disconnect_clients). The reason for adding this is to simply web server hooks (see added docs.)
- Corrects the @return in the docs for disconnect_clients. Also added specs for the existing behavior.
- Updates documentation related to web server forking.

* Fix method name

* More terse syntax

* Preserve old return type

* Update configuration.txt

* Update configuration.txt
  • Loading branch information
johnnyshields authored Apr 22, 2024
1 parent ca3e02e commit aef7114
Show file tree
Hide file tree
Showing 5 changed files with 141 additions and 64 deletions.
109 changes: 48 additions & 61 deletions docs/reference/configuration.txt
Original file line number Diff line number Diff line change
Expand Up @@ -863,58 +863,57 @@ as the following example shows:
Usage with Forking Servers
==========================

When using Mongoid with a forking web server such as Puma, Unicorn or
Passenger, it is recommended to not perform any operations on Mongoid models
in the parent process prior to the fork.

When a process forks, Ruby threads are not transferred to the child processes
and the Ruby driver Client objects lose their background monitoring. The
application will typically seem to work just fine until the deployment
state changes (for example due to network errors, a maintenance event) at
which point the application is likely to start getting ``NoServerAvailable``
exception when performing MongoDB operations.

If the parent process needs to perform operations on the MongoDB database,
reset all clients in the workers after they forked. How to do so depends
on the web server being used.

If the parent process does not need to perform operations on the MongoDB
database after child processes are forked, close the clients in the parent
prior to forking children. If the parent process performs operations on a Mongo
client and does not close it, the parent process will continue consuming a
connection slot in the cluster and will continue monitoring the cluster for
as long as the parent remains alive.

.. note::

The close/reconnect pattern described here should be used with Ruby driver
version 2.6.2 or higher. Previous driver versions did not recreate
monitoring threads when reconnecting.
When using Mongoid with a forking web server such as Puma, or any application
that otherwise forks to spawn child processes, special considerations apply.

If possible, we recommend to not perform any MongoDB operations in the parent
process prior to forking, which will avoid any forking-related pitfalls.

A detailed technical explanation of how the Mongo Ruby Driver handles forking
is given in the `driver's "Usage with Forking Servers" documentation
<https://www.mongodb.com/docs/ruby-driver/current/reference/create-client/#usage-with-forking-servers>`.
In a nutshell, to avoid various connection errors such as ``Mongo::Error::SocketError``
and ``Mongo::Error::NoServerAvailable``, you must do the following:

1. Disconnect MongoDB clients in the parent Ruby process immediately *before*
forking using ``Mongoid.disconnect_clients``. This ensures the parent and child
process do not accidentally reuse the same sockets and have I/O conflicts.
Note that ``Mongoid.disconnect_clients`` does not disrupt any in-flight
MongoDB operations, and will automatically reconnect when you perform new
operations.
2. Reconnect your MongoDB clients in the child Ruby process immediately *after*
forking using ``Mongoid.reconnect_clients``. This is required to respawn
the driver's monitoring threads in the child process.

Most web servers provide hooks that can be used by applications to
perform actions when the worker processes are forked. The following
are configuration examples for several common Ruby web servers.

Puma
----

Use the ``on_worker_boot`` hook to reconnect clients in the workers and
the ``before_fork`` hook to close clients in the parent process
(`Puma documentation <https://puma.io/puma/>`_):
the ``before_fork`` and ``on_refork`` hooks to close clients in the
parent process (`Puma documentation <https://puma.io/puma/#clustered-mode>`_).

.. code-block:: ruby

on_worker_boot do
if defined?(Mongoid)
Mongoid::Clients.clients.each do |name, client|
client.close
client.reconnect
end
else
raise "Mongoid is not loaded. You may have forgotten to enable app preloading."
end
end
# config/puma.rb

# Runs in the Puma master process before it forks a child worker.
before_fork do
if defined?(Mongoid)
Mongoid.disconnect_clients
end
Mongoid.disconnect_clients
end

# Required when using Puma's fork_worker option. Runs in the
# child worker 0 process before it forks grandchild workers.
on_refork do
Mongoid.disconnect_clients
end

# Runs in each Puma child process after it forks from its parent.
on_worker_boot do
Mongoid.reconnect_clients
end

Unicorn
Expand All @@ -926,21 +925,14 @@ the ``before_fork`` hook to close clients in the parent process

.. code-block:: ruby

after_fork do |server, worker|
if defined?(Mongoid)
Mongoid::Clients.clients.each do |name, client|
client.close
client.reconnect
end
else
raise "Mongoid is not loaded. You may have forgotten to enable app preloading."
end
# config/unicorn.rb

before_fork do |_server, _worker|
Mongoid.disconnect_clients
end

before_fork do |server, worker|
if defined?(Mongoid)
Mongoid.disconnect_clients
end
after_fork do |_server, _worker|
Mongoid.reconnect_clients
end

Passenger
Expand All @@ -956,12 +948,7 @@ before the workers are forked.

if defined?(PhusionPassenger)
PhusionPassenger.on_event(:starting_worker_process) do |forked|
if forked
Mongoid::Clients.clients.each do |name, client|
client.close
client.reconnect
end
end
Mongoid.reconnect_clients if forked
end
end

Expand Down
10 changes: 10 additions & 0 deletions lib/mongoid.rb
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,16 @@ def disconnect_clients
Clients.disconnect
end

# Reconnect all active clients.
#
# @example Reconnect all active clients.
# Mongoid.reconnect_clients
#
# @return [ true ] True.
def reconnect_clients
Clients.reconnect
end

# Convenience method for getting a named client.
#
# @example Get a named client.
Expand Down
16 changes: 13 additions & 3 deletions lib/mongoid/clients.rb
Original file line number Diff line number Diff line change
Expand Up @@ -47,9 +47,19 @@ def default
#
# @return [ true ] True.
def disconnect
clients.values.each do |client|
client.close
end
clients.each_value(&:close)
true
end

# Reconnect all active clients.
#
# @example Reconnect all active clients.
# Mongoid::Clients.reconnect
#
# @return [ true ] True.
def reconnect
clients.each_value(&:reconnect)
true
end

# Get a stored client with the provided name. If no client exists
Expand Down
44 changes: 44 additions & 0 deletions spec/mongoid/clients_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1216,4 +1216,48 @@ class StoreChild2 < StoreParent
end
end
end

context "#disconnect" do

let(:clients) do
Mongoid::Clients.clients.values
end

before do
Band.all.entries
end

it "disconnects from all active clients" do
clients.each do |client|
expect(client).to receive(:close).and_call_original
end
Mongoid::Clients.disconnect
end

it "returns true" do
expect(Mongoid::Clients.disconnect).to eq(true)
end
end

context "#reconnect" do

let(:clients) do
Mongoid::Clients.clients.values
end

before do
Band.all.entries
end

it "reconnects all active clients" do
clients.each do |client|
expect(client).to receive(:reconnect).and_call_original
end
Mongoid::Clients.reconnect
end

it "returns true" do
expect(Mongoid::Clients.reconnect).to eq(true)
end
end
end
26 changes: 26 additions & 0 deletions spec/mongoid_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,32 @@
end
Mongoid.disconnect_clients
end

it "returns true" do
expect(Mongoid.disconnect_clients).to eq(true)
end
end

describe ".reconnect_clients" do

let(:clients) do
Mongoid::Clients.clients.values
end

before do
Band.all.entries
end

it "reconnects all active clients" do
clients.each do |client|
expect(client).to receive(:reconnect).and_call_original
end
Mongoid.reconnect_clients
end

it "returns true" do
expect(Mongoid.reconnect_clients).to eq(true)
end
end

describe ".client" do
Expand Down

0 comments on commit aef7114

Please sign in to comment.