diff --git a/build-all-multiplatform.sh b/build-all-multiplatform.sh index ba1b7223..46bf8b93 100755 --- a/build-all-multiplatform.sh +++ b/build-all-multiplatform.sh @@ -29,17 +29,12 @@ 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_VERSIONS="3.2" -MONGO_MULTIPLATFORM_VERSIONS="3.4 3.6 4.0 4.2 4.4 5.0 6.0" +MONGO_MULTIPLATFORM_VERSIONS="5.0 6.0 7.0" -POSTGRES_VERSIONS="9.6 10 11 12 13 14" -POSTGRES_MULTIPLATFORM_VERSIONS="15 16 17 9.6-alpine 10-alpine 11-alpine 12-alpine 13-alpine 14-alpine 15-alpine 16-alpine 17-alpine" +POSTGRES_VERSIONS="12 13 14" +POSTGRES_MULTIPLATFORM_VERSIONS="15 16 17 12-alpine 13-alpine 14-alpine 15-alpine 16-alpine 17-alpine" if shouldBuild "mongo"; then - for MONGO_VERSION in $MONGO_VERSIONS; do - ./build-mongo-multiplatform.sh "$MONGO_VERSION" "linux/amd64" - done - for MONGO_VERSION in $MONGO_MULTIPLATFORM_VERSIONS; do ./build-mongo-multiplatform.sh "$MONGO_VERSION" "${MULTIPLATFORM_PLATFORMS}" done diff --git a/build-all.sh b/build-all.sh index b8eb864a..81af1945 100755 --- a/build-all.sh +++ b/build-all.sh @@ -6,7 +6,7 @@ DEBEZIUM_VERSION="3.0" if [ -z "$DEBEZIUM_VERSIONS" ]; then DEBEZIUM_VERSIONS="$DEBEZIUM_VERSION" fi -MONGO_VERSIONS="3.2 3.4 3.6 4.0 4.2 4.4 5.0 6.0" +MONGO_VERSIONS="5.0 6.0 7.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 diff --git a/mongo-initiator/7.0/Dockerfile b/mongo-initiator/7.0/Dockerfile new file mode 100644 index 00000000..5602158e --- /dev/null +++ b/mongo-initiator/7.0/Dockerfile @@ -0,0 +1,10 @@ +FROM mongo:7.0 + +LABEL maintainer="Debezium Community" + +COPY ./docker-entrypoint.sh / + +USER mongodb + +ENTRYPOINT ["/docker-entrypoint.sh"] +CMD ["start"] diff --git a/mongo-initiator/7.0/README.md b/mongo-initiator/7.0/README.md new file mode 100644 index 00000000..8fe3b980 --- /dev/null +++ b/mongo-initiator/7.0/README.md @@ -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 + + + diff --git a/mongo-initiator/7.0/docker-entrypoint.sh b/mongo-initiator/7.0/docker-entrypoint.sh new file mode 100755 index 00000000..f2c2f621 --- /dev/null +++ b/mongo-initiator/7.0/docker-entrypoint.sh @@ -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 "$@"