Skip to content

Commit

Permalink
DBZ-8451 Support Mongo 8.0
Browse files Browse the repository at this point in the history
  • Loading branch information
ani-sha committed Dec 2, 2024
1 parent dcb700d commit 8d22b15
Show file tree
Hide file tree
Showing 5 changed files with 215 additions and 2 deletions.
2 changes: 1 addition & 1 deletion build-all-multiplatform.sh
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ function shouldBuild() {
# with multi platform build
DEBEZIUM_SINGLEPLATFORM_VERSIONS="1.0 1.1 1.2 1.3 1.4 1.5 1.6 1.7 1.8"

MONGO_MULTIPLATFORM_VERSIONS="5.0 6.0 7.0"
MONGO_MULTIPLATFORM_VERSIONS="5.0 6.0 7.0 8.0"

POSTGRES_VERSIONS="12 13 14"
POSTGRES_MULTIPLATFORM_VERSIONS="15 16 17 12-alpine 13-alpine 14-alpine 15-alpine 16-alpine 17-alpine"
Expand Down
2 changes: 1 addition & 1 deletion build-all.sh
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ DEBEZIUM_VERSION="3.0"
if [ -z "$DEBEZIUM_VERSIONS" ]; then
DEBEZIUM_VERSIONS="$DEBEZIUM_VERSION"
fi
MONGO_VERSIONS="5.0 6.0 7.0"
MONGO_VERSIONS="5.0 6.0 7.0 8.0"
POSTGRES_VERSIONS="12 12-alpine 13 13-alpine 14 14-alpine 15 15-alpine 16 16-alpine 17 17-alpine"

for MONGO_VERSION in $MONGO_VERSIONS; do
Expand Down
10 changes: 10 additions & 0 deletions mongo-initiator/8.0/Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
FROM mongo:8.0

LABEL maintainer="Debezium Community"

COPY ./docker-entrypoint.sh /

USER mongodb

ENTRYPOINT ["/docker-entrypoint.sh"]
CMD ["start"]
26 changes: 26 additions & 0 deletions mongo-initiator/8.0/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
Initiate a MongoDB replica set on a set of running MongoDB servers, and optionally add the replica set as a shard to one or more running MongoDB routers.

# How to use this image

## Initiate a replica set

With one or more MongoDB servers already running, start a container with this image to check if the replica set is initiated, and if not to initiate it and add all of the servers to the replica set. Start the container with the name of the replica set in the `$REPLICASET` environment variable, using links named like "MONGO_n_" (where _n_=1,2,3, etc.) for each of the MongoDB servers that are to be in the replica set.

For example, consider three already-running MongoDB servers running in containers `data1`, `data2`, and `data3`. Running a container using the following command will check whether the replica set named `rs0` is properly initiated on these servers, and if not will initiate the replica set:

$ docker run -it --name mongo-init --rm -e REPLICASET=rs0 --link data1:mongo1 --link data2:mongo2 --link data3:mongo3 quay.io/debezium/mongo-initiator

The container will exit as soon as the replica set is initiated.

## Initiate a shard replica set

The container can optionally add the replica set as a _shard_ to one or more MongoDB routers. For example, consider three MongoDB servers running in containers `shardA1`, `shardA2`, and `shardA3`, and two MongoDB routers running in containers `router1` and `router2`. The following command will ensure that `shardA1`, `shardA2`, and `shardA3` are properly initiated as replica set `shardA`, and that the `shardA` replica set is added as a shard to the routers `router1` and `router2`:

$ docker run -it --name mongo-init --rm -e REPLICASET=shardA --link shardA1:mongo1 --link shardA2:mongo2 --link shardA3:mongo3 --link router1 --link router2 quay.io/debezium/mongo-initiator

Additional shard replica sets can be initiated and added by running additional containers. For example:

$ docker run -it --name mongo-init --rm -e REPLICASET=shardB --link shardB1:mongo1 --link shardB2:mongo2 --link shardB3:mongo3 --link router1 --link router2 quay.io/debezium/mongo-initiator



177 changes: 177 additions & 0 deletions mongo-initiator/8.0/docker-entrypoint.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,177 @@
#!/bin/bash

# Exit immediately if a *pipeline* returns a non-zero status. (Add -x for command tracing)
set -Eo pipefail

# The names of container links are as follows, where 'n' is 1, 2, 3, etc.:
#
# "MONGOn" - specifies the Mongo replica set nodes, and n=1 for the primary node
# "ROUTERn" - specifies the Mongo router servers (optional, order is not important)
#
# The following environment variables are recognized:
#
# "REPLICASET" - specifies the name of the replica set for storing data/shards
#
# This script will attempt to initiate the replica set in the Mongo servers specified in the one or more `MONGOn`
# links, where the name of the replica set is given in the `REPLICASET` environment variable. (This step does
# nothing if the replica set is already initiated.) Then, if one or more `ROUTERn` links are specified, this
# script will add the replica set named `REPLICASET` as a shard to each of these MongoDB routers.
#

if [[ -z $1 ]]; then
ARG1="start"
else
ARG1=$1
fi

MONGO="/usr/bin/mongosh"

#
# Process all MongoDB nodes by looking for environment variables that match 'MONGOn_PPORT_*_ADDR':
#
NODE_COUNT=0;
for VAR in `env | sort`
do
# First look for port 27019 used for the config server replica sets
env_var=`echo "$VAR" | sed -r "s/(.*)=.*/\1/g"`
if [[ $env_var =~ ^MONGO([0-9]+)_PORT_27019_TCP_ADDR ]]; then
NODE_COUNT=$(($NODE_COUNT+1))
NODE_HOSTS[$NODE_COUNT]="${!env_var}";
NODE_PORTS[$NODE_COUNT]="27019";
if [[ $NODE_COUNT -eq 1 ]]; then
PRIMARY_HOST=${!env_var}
PRIMARY_PORT=27019
fi
fi
done
if [[ -z $PRIMARY_HOST ]]; then
NODE_COUNT=0;
# None were found, so use whatever is available ...
for VAR in `env | sort`
do
env_var=`echo "$VAR" | sed -r "s/(.*)=.*/\1/g"`
if [[ $env_var =~ ^MONGO([0-9]+)_PORT_([0-9]+)_TCP_ADDR ]]; then
NODE_COUNT=$(($NODE_COUNT+1))
NODE_HOSTS[$NODE_COUNT]="${!env_var}";
NODE_PORTS[$NODE_COUNT]="${BASH_REMATCH[2]}";
if [[ $NODE_COUNT -eq 1 ]]; then
PRIMARY_HOST=${!env_var}
PRIMARY_PORT=${BASH_REMATCH[2]}
fi
fi
done
fi

#
# Process all MongoDB *router* nodes by looking for environment variables that match 'ROUTERn_PPORT_*_ADDR':
#
ROUTER_COUNT=0;
for VAR in `env | sort`
do
env_var=`echo "$VAR" | sed -r "s/(.*)=.*/\1/g"`
if [[ $env_var =~ ^ROUTER([0-9]+)_PORT_([0-9]+)_TCP_ADDR ]]; then
ROUTER_COUNT=$(($ROUTER_COUNT+1))
ROUTER_HOSTS[$ROUTER_COUNT]="${!env_var}";
ROUTER_PORTS[$ROUTER_COUNT]="${BASH_REMATCH[2]}";
fi
done

export NODE_HOSTS
export NODE_PORTS
export ROUTER_HOSTS
export ROUTER_PORTS

# Process some known arguments ...
case $ARG1 in
start)
if [[ $NODE_COUNT -lt 1 ]]; then
echo "At least one Mongo node must be defined in container links, starting with \"MONGO1\" for the primary node and continuing with \"MONGO2\", etc. for secondaries."
exit 1;
fi
if [[ -z "$REPLICASET" ]]; then
echo "The name of the replica set must be defined with the \"REPLICASET\" environment variable"
exit 1;
fi

# Define the command that will connect the mongo shell to the primary node ...
PRIMARY_MONGO="$MONGO --host $PRIMARY_HOST --port $PRIMARY_PORT"

# Wait for the nodes to become available ...
echo "Testing connection to MongoDB primary node at ${PRIMARY_HOST}:${PRIMARY_PORT} ..."
status=$($PRIMARY_MONGO --eval db --quiet)
if [ "$?" -ne 0 ]; then
echo "Error: Unable to connect to ${PRIMARY_HOST}:${PRIMARY_PORT}"
echo ""
echo "${status}"
exit 2
fi

# See if the replica set is not set up ...
rsStatus=$($PRIMARY_MONGO --eval "rs.status()" --quiet 2>&1)
if [[ $rsStatus =~ "no replset config has been received" ]]; then
# Set up the replica set configuration document ...
echo "Using MongoDB primary node to initiate replica set \"${REPLICASET}\" with:"
CONFIGVAR="config= {_id: \"${REPLICASET}\", members:[ "
hostNum=1
while [[ $hostNum -le $NODE_COUNT ]]; do
hostAndPort="${NODE_HOSTS[$hostNum]}:${NODE_PORTS[$hostNum]}"
if [[ $hostNum -eq 1 ]]; then
priority=2
echo "- primary node: ${hostAndPort}"
else
priority=1
echo "- secondary node: ${hostAndPort}"
CONFIGVAR="${CONFIGVAR},"
fi
CONFIGVAR="${CONFIGVAR} {_id: ${hostNum}, host: \"${hostAndPort}\", priority: ${priority} }"
hostNum=$hostNum+1
done
CONFIGVAR="${CONFIGVAR} ] }"

# Initiate the replica set with our document ...
$PRIMARY_MONGO --eval "${CONFIGVAR};rs.initiate(config);" --quiet

rsStatus=$($PRIMARY_MONGO --eval "rs.status()" --quiet 2>&1)
if [[ $rsStatus =~ "no replset config has been received" ]]; then
echo "Failed to initialize replica set"
exit 1
fi
else
echo "Replica set \"${REPLICASET}\" is already initiated."
fi
#if [[ -n $VERBOSE ]]; then
# echo ""
# echo "Current replica status:"
# echo "${rsStatus}"
#fi
echo ""
echo "Replica set is ready"
echo ""

if [[ $ROUTER_COUNT -gt 0 ]]; then
# Add the primary of the replica set as a shard to each of the routers ...
echo ""
echo "Checking ${ROUTER_COUNT} router for shard using replica set \"${REPLICASET}/${PRIMARY_HOST}:${PRIMARY_PORT}\":"
hostNum=1
added=0
while [[ $hostNum -le $ROUTER_COUNT ]]; do
hostAndPort="${ROUTER_HOSTS[$hostNum]}:${ROUTER_PORTS[$hostNum]}"
result=$($MONGO --host ${ROUTER_HOSTS[$hostNum]} --port ${ROUTER_PORTS[$hostNum]} --eval "rs.status();sh.addShard( \"${REPLICASET}/${PRIMARY_HOST}:${PRIMARY_PORT}\" )" --quiet 2>&1)
if [[ $result =~ "E11000 duplicate key error collection: config.shards index: _id_ dup key" ]]; then
echo "- ${hostAndPort} (shard exists)"
else
added=$(($added+1))
echo "- ${hostAndPort} (adding shard)"
fi
hostNum=$hostNum+1
done
echo "Added replica set \"${REPLICASET}/${PRIMARY_HOST}:${PRIMARY_PORT}\" as shard to ${added} routers."
echo ""
echo "Routers are ready."
fi
exit 0
;;
esac

# Otherwise just run the specified command
exec "$@"

0 comments on commit 8d22b15

Please sign in to comment.