I was recently tasked with setting up a MongoDB Replicaset.

Documentation for setting up a MongoDB Replicaset with OpsWorks was a bit on the sparse side and took a bit of trial and error. I ended up with this solution.

The Chef Recipe

The chef recipe used for this post is https://github.com/netinlet/chef-mongodb_replicaset

OpsWorks allows you to use custom recipes. All dependent recipes should be included in the custom cookbook.

The mongodb recipe depends on apt, python, runit and yum recipes. Several recipes require the build-essentials recipes and the runit recipe depends upon yum-epl.

The dependencies for this cookbook were pulled from the OpsCode public cookbooks. https://github.com/opscode-cookbooks

The mongodb recipe is custom. I ended up using this recipe referenced in the presenation AWS OpsWorks under the hood.

The custom recipe is based off of https://github.com/edelight/chef-mongodb with OpsWorks specific code to properly setup replicasets in AWS OpsWorks.

OpsWorks Setup

I’m going to assume some degree of familiarity with OpsWorks. If you are new to AWS OpsWorks, you can get an overview here: http://aws.amazon.com/opsworks/

Add a MongoDB layer to your stack

Create a “Custom” layer named MongoDB and add it to the stack.

Configure the MongoDB layer

A custom layer is pre-defined with 12 built-in recipes. Nothing to do or change here.

Custom Chef Recipe Section

Repository URL Set the repository URL to the location of your chef recipe. This repository can be a link to a Git or Subversion repository or an http or S3 url. The custom recipe is configured in the Stack settings in which your layer lives. In this case, I have it pointed to my Github chef recipe.

OpsWorks has 5 lifecycle events which can be used to execute Chef recipe steps. These events are setup, configure, deploy, undeploy and shutdown. For this recipe, we will use the setup and configure events.

Setup event Add mongodb::10gen_repo and mongodb::default. The former configures the service to use the 10Gen MongoDB builds. The latter installs MongoDB and does a basic single node configuration.

Configure event Add mongodb::replicaset. This event configures the MongoDB instance as a member of a replicaset.

Custom Chef JSON In order for the replicaset to be configured properly, we have to specify the name of the replicaset, cluster and tell the instance to auto-configure.

The replicaset_name and cluster_name must match. You must also set the auto_configure:replicaset hash value to true so the replicaset is auto-configured on boot.

{
  "mongodb" : {
    "replicaset_name": "mongo-rs",
      "cluster_name": "mongo-rs",
      "auto_configure": {
        "replicaset": true
      }
  }
}

Screenshot of editing the Stack where the custom recipe and Chef JSON are configured

Boot your MongoDB Instances

Add some instances to the MongoDB layer and boot them. An odd number of instances (3 or more) is the proper way to configure a replicaset without an arbiter.

Unfortunately, on first boot the primary replicaset member is unable to communicate with the secondary replicaset member. I was unable to determine exactly why this happens but suspect a timing issue with the availability of the secondary replicaset member.

This is evident in the logs of the second instance. This error indicates the node was unreachable at the time it was being configured. See line 4.

[2014-01-12T23:15:37+00:00] INFO: ruby_block[run_config_replicaset] sending create action to ruby_block[config_replicaset] (delayed)
[2014-01-12T23:15:37+00:00] INFO: Processing ruby_block[config_replicaset] action create (mongodb::replicaset line 178)
[2014-01-12T23:15:42+00:00] INFO: Configuring replicaset with members adrastea, venus
[2014-01-12T23:15:42+00:00] ERROR: configuring replicaset returned: #<BSON::OrderedHash:0x3fa7491cb9f4 {"errmsg"=>"exception: need most members up to reconfigure, not ok : ip-172-31-0-241.ec2.internal:27017", "code"=>13144, "ok"=>0.0}>
[2014-01-12T23:15:42+00:00] INFO: ruby_block[config_replicaset] called

The work-around is to run the mongodb::replicaset recipe again from the Stack section of OpsWorks with the Run Command button.

*Note: MongoDB instance names changed because I removed the nodes before realizing the original screenshot did not get saved properly.

Verifying the configuration

If everything was successful, you should be able to ssh to your MongoDB instances and look at the MongoDB configuration with the commands mongo to connect to the instance and rs.conf() to show the current replicaset configuration.

This screenshot shows the before and after of the above mentioned mongodb::replicaset command work-around.