I recently got the opportunity to pick up development on a Ruby on Rails application that was originally setup to run on AWS using their Elastic Beanstalk deployment tools. One of our first tasks was to move some notification hooks out of the normal workflow into scripts and schedule those batch scripts using cron.
Historically, I've had extremely good luck with Whenever. In my previous endeavors I've utilized Capistrano which Whenever merged with seamlessly. With how simple it was to integrate Whenever with Capistrano, I anticipated a similar experience dealing with Elastic Beanstalk. While the integration was not as seamless as Capistrano, I did manage to make it work.
My first stumbling block was finding documentation on how to do after or post hooks. I managed to find this forum post and this blog post which helped me out a lot. The important detail is that there is a "post" directory to go along with "pre" and "enact", but it's not present by default, so it can be easy to miss.
I used Marcin's delayed_job config as a base. The first thing I had to address was an apparent change in Elastic Beanstalk's configuration structure. Marcin's config has
. /opt/elasticbeanstalk/support/envvarsbut that file doesn't exist on the system I was working on. With a small amount of digging, I found:
. /opt/elasticbeanstalk/containerfiles/envvarsin one of the other ebextensions. Inspecting that file showed a definition for and exportation of
$EB_CONFIG_APP_CURRENTsuggesting this is a similar file just stored in a different location now.
Another change that appears to have occurred since Marcin developed his config is that directories will be created automatically if they don't already exist when adding a file in the files section of the config. That allows us to remove the entire commands section to simplify things.
That left me with a config that looked like:
files: "/opt/elasticbeanstalk/hooks/appdeploy/post/99_update_cron.sh" mode: "000755" owner: root group: root content: | #! /usr/bin/env bash . /opt/elasticbeanstalk/containerfiles/envvars su -c "cd $EB_CONFIG_APP_CURRENT; bundle exec whenever --update-cron" - $EB_CONFIG_APP_USER
This command completed successfully but on staging the cron jobs failed to run. The reason for that was an environment mismatch. The runner entries inside the cron commands weren't receiving a RAILS_ENV or other type of environment directive so they were defaulting to production and failing when no database was found.
After some greping I was able to find a definition for RACK_ENV in:
/opt/elasticbeanstalk/containerfiles/envvars.d/sysenvMaking use of it, I came up with this final version:
files: "/opt/elasticbeanstalk/hooks/appdeploy/post/99_update_cron.sh" mode: "000755" owner: root group: root content: | #! /usr/bin/env bash . /opt/elasticbeanstalk/containerfiles/envvars . /opt/elasticbeanstalk/containerfiles/envvars.d/sysenv su -c "cd $EB_CONFIG_APP_CURRENT; bundle exec whenever --update-cron --set='environment=$RACK_ENV'" - $EB_CONFIG_APP_USER