Recently, I got a Rails app running on AWS Elastic Beanstalk for the first time. I wanted to share the steps to do so here. Although there are other tutorials, AWS and Rails both change over time, so some of these steps here are updated compared to other tutorials.
Since AWS is so complex, a lot of tutorials assume prior AWS knowledge as well; this one walks you through everything you need step by step.
In this tutorial, we’ll cover how to:
- Get an Elastic Beanstalk instance running the default sample Ruby application, including logging
- Create a fresh “hello world” Rails app
- Configure the Rails app to automatically deploy to Elastic Beanstalk using AWS CodePipeline
- Optionally, set up a custom domain for the app with NameCheap
- Optionally, set up SSL for the app
I say “optionally” for the last two steps because if you aren’t ready to register a custom domain or set up SSL, you can stop before that point and the application will still work.
Note that we won’t be covering database setup. This is just because I didn’t run across that myself because I’m using a pre-existing database on another service. Amazon does have an RDS service for providing relational databases like Postgres and MySQL. Also, although the demo app here is a Rails server-rendered app, these steps would work the same for Rails APIs such as REST or GraphQL.
Note that this doesn’t cover more advanced server configurations such as multiple instances, test and staging environments, and zero-downtime deployments.
Another important note: AWS costs money. The instance size we need to use for Rails is too big for the free tier, so you’ll be charged for it, and there’s a limit to how many build minutes can be run in the free tier as well. So keep an eye on your usage, and spin down the application if you aren’t using it.
This tutorial was written in February 2024 using Ruby 3.2 and Rails 7.1.3.
Getting an Elastic Beanstalk instance running
Go to the AWS Management Console and sign in or sign up. In the Search box, type “Elastic Beanstalk” and click the Elastic Beanstalk link.
data:image/s3,"s3://crabby-images/824f1/824f1295b50dbb0a8c98d3efc2e5584d0cda12bb" alt="The AWS Console search box, with Elastic Beanstalk selected."
From the Elastic Beanstalk home page, click “Create application.”
data:image/s3,"s3://crabby-images/c48d1/c48d150984cd6a8e3a33d402e09f833a8bed1cfc" alt="The AWS Elastic Beanstalk home page."
You’ll be taken to the “Configure environment” screen. Under “Environment tier,” make sure “Web server environment” is selected. Under “Application information,” type a name for your application; I’ll do “rails-eb”.
data:image/s3,"s3://crabby-images/a535b/a535bcc63600f6a8c8b4dc8c25bc89175db665e2" alt="The 'Configure Environment' step of creating a new Elastic Beanstalk application. 'rails-eb' has been entered as the Application Name."
Under “Platform” > “Platform,” choose Ruby.
data:image/s3,"s3://crabby-images/fb1f0/fb1f043e28b885a4fadc09a66991309daaf496f8" alt="The 'Platform' selection box of the Elastic Beanstalk application creation wizard. Ruby is selected as the platform."
Under “Application code,” make sure “Sample application” is chosen; we’ll use the pre-provided Ruby sample first, then set up uploads via CodePipeline later.
data:image/s3,"s3://crabby-images/1e145/1e1459aa4baa58811728e6f5f9a79cf7824f3848" alt="The 'Application code' step of the Elastic Beanstalk application creation wizard. 'Sample application' is selected."
Click “Next.”
You’ll see the “Configure service access” screen. You may see some pre-existing service roles preselected; if not, choose “Create and use new service role.”
data:image/s3,"s3://crabby-images/978ee/978ee72413c2771ddd35d62e5fbeacb62ffbcc98" alt="The 'Configure service access' step of the Elastic Beanstalk application creation wizard."
Click “Next.”
On the “Set up networking, database, and tags” screen, scroll to the bottom and click “Next.”
On the “Configure instance traffic and scaling” screen, scroll down to “Capacity,” then find “Instance types.” You should see “t3.micro” and “t3.small” listed by default. Click the X next to “e3.micro” to remove it so that only “t3.small” is shown. When I tried to run a Rails app on a “t3.micro” instance, the bundle install
step would hang, so I found that “t3.small” or larger is needed.
data:image/s3,"s3://crabby-images/02ad9/02ad92bd5ddc7c181e9df568aa270d73233664c7" alt="The 'Instance types' section of the Elastic Beanstalk application creation wizard. Only 't3.small' is selected as an instance type."
Click “Next.”
On “Configure updates, monitoring, and logging,” scroll down to “Platform software.” Under “Instance log streaming to CloudWatch logs,” set “Log streaming” to “Activated.” (Note that this is different from “Health event streaming to CloudWatch logs” further up on the page.)
data:image/s3,"s3://crabby-images/becd9/becd9b939de7a555eeb11922bfe3d9092a4a77f6" alt="The 'Platform software' section of the Elastic Beanstalk application creation wizard. Under 'Log streaming,' the 'Activated' checkbox is checked."
Under “Environment properties,” we will need to add a SECRET_KEY_BASE
value for Rails. One way to generate one is to run irb
in your console and run the following commands:
irb(main):001> require 'securerandom'
=> true
irb(main):002> SecureRandom.hex(64)
=> "(a 64-character string)"
Under “Environment properties,” click “Add environment property,” then type “SECRET_KEY_BASE” in the “Name” column and paste the secret key base value you generated into the Value column.
data:image/s3,"s3://crabby-images/5bd0d/5bd0db5f6f3d54d541de8ed430bc56840e41fe00" alt="The 'Environment properties' section of the Elastic Beanstalk application creation wizard. A SECRET_KEY_BASE property has been added."
Click “Next.”
On the “Review” screen, scroll down and click “Submit.”
You’ll be taken to the screen for the new environment that was created. Yours will probably be the name of your application with “-env” on the end. Mine has a “-1” as well (“Rails-eb-env-1”) because I previously created a “Rails-eb-env” while writing this tutorial.
You’ll see a message that says: “Elastic Beanstalk is launching your environment. This will take a few minutes.”
data:image/s3,"s3://crabby-images/8683f/8683f590ce64e22d316c6b532f14f0aedf4b93ae" alt="The 'Environment' page of Elastic Beanstalk, showing an environment named Rails-eb-env-1. A message says that Elastic Beanstalk is launching your environment."
You can watch the launch process at the bottom of the page under Events. Eventually, you should see the message “Environment successfully launched.”
data:image/s3,"s3://crabby-images/26b39/26b39dbdb2a9da66217635d85992ea3c81b069d8" alt="The 'Environment' page of Elastic Beanstalk, showing an environment named 'Rails-eb-env-1'. A message says that the environment was successfully launched."
At the top of the page, under Environment overview > Domain, you’ll see a URL ending with .elasticbeanstalk.com
.
data:image/s3,"s3://crabby-images/a28d6/a28d675895e175afd410093748789c0b79c23867" alt="The 'Environment' page of Elastic Beanstalk, showing an environment named Rails-eb-env-1. Under 'Environment overview,' a domain is visible starting with the name of the environment."
Click it, and you should see a page that says: “Congratulations. Your first AWS Elastic Beanstalk Ruby Application is now running on your own dedicated environment in the AWS Cloud.”
data:image/s3,"s3://crabby-images/18f6a/18f6af3e22318ea844bb5e14db44af4765efbb99" alt="A web page with a message that says: 'Congratulations. Your first AWS Elastic Beanstalk Ruby Application is now running on your own dedicated environment in the AWS Cloud."
Let’s confirm that our logs are working too. At the top of the page in the search box, type “CloudWatch”.
data:image/s3,"s3://crabby-images/eca6a/eca6ab45bc8cf35a16c26968ed47c669ad6684fd" alt="The AWS search box, with CloudWatch selected."
Open the CloudWatch link in a new browser tab so that your Elastic Beanstalk tab stays open too.
In CloudWatch, click “Logs” > “Log groups,” then look for a line that includes the name of your environment—in my case, it added “-env” to the name of my app, which was “Rails-eb”. You should see several files for that environment; look for the one ending in “/eb-engine.log” and click it.
data:image/s3,"s3://crabby-images/cb418/cb418e8633799e56cc1289c1133477bd82b2a694" alt="The CloudWatch 'Log groups' page, showing a series of log groups for Rails-eb-env-1."
(If you don’t see it, you may have missed the “Instance log streaming to CloudWatch logs” checkbox in application setup; if so, you can go to your “Environment” under Elastic Beanstalk, then “Configuration,” then “Updates, monitoring, and logging,” click “Edit,” and activate those logs.)
On the eb-engine.log page, go to “Log streams,” then click the link you see.
data:image/s3,"s3://crabby-images/313d3/313d3a84ec8dab965d8b0ffb15364468f26555b6" alt="A 'Log groups' page in CloudWatch, showing a list of 'Log streams.'"
You should see a log of deployment output.
data:image/s3,"s3://crabby-images/afa4e/afa4ed0056ad29b9b1170d0452e3a988da9aba0d" alt="A 'Log groups' page in CloudWatch, showing log output ending with 'Platform Engine finished execution.'"
This will be helpful to watch for future deployments; leave this browser tab open.
Creating a Sample Rails App
So, we have an Elastic Beanstalk instance running the sample code. Next, let’s get it running our own code.
To do so, let’s create a sample Rails app to run. You may be tempted to use your real Rails app, but I’d encourage creating the sample app first instead. Your app may need additional setup that can cause errors, so let’s take the agile small step of deploying a trivial app first. I’ll keep it quick!
First, we need to make sure we’re running a version of Ruby that runs on Elastic Beanstalk. Go to Elastic Beanstalk’s Supported Platforms page and check the version of Ruby listed; it’s 3.2.2 as of this writing.
Next, check the version of Ruby you have running locally. Here’s the result for me:
$ ruby -v
ruby 3.2.2 (2023-03-30 revision e51014f9c0) [arm64-darwin22]
If your version of Ruby is newer than the one Amazon lists, install 3.2.2 via a tool like rbenv
.
Once you have the right Ruby version, run rails -v
and make sure you have Rails 7.1.3 or later installed. If not, run gem update rails
.
Next, create a new Rails application with all the defaults:
$ rails new rails-eb
After the new command is done, create a welcome page:
$ cd rails-eb
$ rails generate controller WelcomePage welcome
Replace the contents of app/views/welcome_page/welcome.html.erb
with:
<h1>Rails on Elastic Beanstalk</h1>
Then, in config/routes.rb
, add the following:
Rails.application.routes.draw do
get 'welcome_page/welcome'
+ root 'welcome_page#welcome'
...
Test this by running your app with rails s
, then going to http://127.0.0.1:3000
. You should see the “Rails on Elastic Beanstalk” message you entered.
data:image/s3,"s3://crabby-images/8e26d/8e26ddb6d681bcf5a28dbd6a215606970414e42d" alt="A web page with only a heading that says: 'Rails on Elastic Beanstalk'"
There’s one more temporary change we need to make. By default, Rails enforces SSL security in production. This is very good, but to make sure we can confirm our app is working before we set up SSL, we’re going to turn that off. If you do this for a real app, make sure you turn SSL back on before you send users to it!
Open config/environments/production.rb
and change config.force_ssl
to false
# Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
-config.force_ssl = true
+config.force_ssl = false # TEMPORARY for testing
Create a GitHub repository for your Rails app and push the code up to it.
Deploy with CodePipeline
Now we’ll set up our Rails code to be deployed to our Elastic Beanstalk instance using CodePipeline.
Open yet another new browser tab and go to the AWS Management Console. Search for “CodePipeline”, then click the CodePipeline link.
data:image/s3,"s3://crabby-images/ed243/ed2434f17fe775cb16ed98069a3f557579aa17d5" alt="The AWS Console search, with CodePipeline selected."
Click “Create pipeline.”
data:image/s3,"s3://crabby-images/1f6ff/1f6ffeaf0a18d86d262978b4e0a4ef119a908293" alt="The CodePipeline Pipelines page."
You’ll see the “Choose pipeline settings” page. Under “Pipeline settings” > “Pipeline name,” enter a name; you can call it the same as your application, which for me is “rails-eb”.
data:image/s3,"s3://crabby-images/0faaa/0faaa577aa537f51f1779fe69814f6545ee09f77" alt="The 'Choose pipeline settings' page of the CodePipeline 'Create pipeline' wizard. 'rails-eb' is entered as the 'Pipeline name.'"
Scroll down to “Service role” and make sure “New service role” is selected; you can keep the default “Role name.”
data:image/s3,"s3://crabby-images/3131c/3131c03d588331fa39152c8a26642645a093e5cd" alt="The 'Choose pipeline settings' page of the CodePipeline 'Create pipeline' wizard. Under 'Service role,' 'New service role' is selected."
Click “Next.”
On the “Add source stage” screen, under “Source provider,” choose “GitHub (Version 2).”
data:image/s3,"s3://crabby-images/f444f/f444f9a32fc99d27d0e999951069024a84911f68" alt="The 'Add source stage' screen of the CodePipeline 'Create pipeline' wizard. 'GitHub (Version 2)' is chosen as the 'Source provider.'"
Click “Connect to GitHub” to sign in to your GitHub account and give AWS access to your sample Rails app repo. Under “Repository name,” choose your Rails app repo. Under “Branch name,” choose the branch to deploy, which is probably “main” unless you changed it.
data:image/s3,"s3://crabby-images/9c2d2/9c2d219a60f54afc7ead4b7192de5531e7124b0e" alt="The 'Add source stage' screen of the CodePipeline 'Create pipeline' wizard. A message says: 'Your GitHub connection is ready to use.'"
Under “Trigger,” choose a “Trigger type” of “Specify filter.” For “Event type” choose “Push,” for “Filter type” choose “Branch,” and under “Branches” > “Include” type “main”.
data:image/s3,"s3://crabby-images/2be20/2be2007ed3b6c6f57995f4cee1d681d5702a38a2" alt="The 'Trigger' section of the CodePipeline 'Create pipeline' wizard. It is configured to trigger on pushes to the main branch."
Click “Next.”
Under “Add build stage,” click “Skip build stage.” This isn’t one of those compiled languages! (Running bundle install
doesn’t count as a build here.) Click “Skip” to confirm that you don’t want to be bothered with compilation.
On the “Add deploy stage” screen, under “Deploy provider” choose “AWS Elastic Beanstalk.” For “Application name” and “Environment name” choose your EB application and environment.
data:image/s3,"s3://crabby-images/56f78/56f7822deeb7413e144429507772d2fea3f2783b" alt="The 'Deploy' step of the CodePipeline 'Create pipeline' wizard. It is configured to deploy to the Elastic Beanstalk environment Rails-eb-env-1."
Click “Next.”
You’ll see a “Review” screen; scroll down and click click “Create pipeline.”
After a few seconds, you’ll be taken to the screen for your new pipeline and see the message “Congratulations! The pipeline (name) has been created.”
data:image/s3,"s3://crabby-images/6d0e7/6d0e7288af27769a1e64cd1556c82a9a930bd3ff" alt="The CodePipeline page for a pipeline named 'rails-eb'. A message says: 'Congratulations! The pipeline rails-eb has been created.'"
You’ll also see a two-part diagram showing the steps of your pipeline, and it will begin executing. Warning! It’s possible your Deploy step will fail. If it succeeds, skip down to Install Success below. Otherwise, keep reading.
If your install fails, click “View details” to see why. I’ve run across the following error message (emphasis mine):
Deployment failed. The provided role does not have sufficient permissions: Failed to deploy application. Service:AWSLogs, Message:User: …/AWSCodePipelineServiceRole-… is not authorized to perform: logs:CreateLogGroup on resource: …/var/log/nginx/access.log:log-stream: because no identity-based policy allows the logs:CreateLogGroup action
If that happens, that means there’s a permissions issue. By default, the role CodePipeline created isn’t granted access to write the logs that we said we wanted created. To fix this, we’ll need to make a change in IAM, AWS’s auth tool.
First, copy the name of the role listed here. Then, in the search bar, search for IAM then open the link in a new tab (just “IAM,” not the “IAM Identity Center”).
In IAM, click “Role.” In the search box under “Roles,” paste the role name you copied from the error message. It should match one result; click it.
Under “Permissions” > “Permissions policies,” click “Add permissions” > “Attach policies.” Search for “CloudWatchLogsFullAccess”, then next to the row that’s shown, click the checkbox and click “Add permissions.” You should be taken back to the role page, and the CloudWatchLogsFullAccess permission should now be listed in the “Permissions policies” box.
With permissions fixed, go back to the browser tab with CodePipeline. In the Deploy box, click “Retry stage.” This can take a little while, and we can watch the process in more detail in CloudWatch logs. Scroll to the bottom and click the “Resume” link to get the logs updated live. After a few seconds, you should see a “Starting…” line. It may stop at the bundle _2.4.10_ install
line for a bit. (If it hangs at bundle _2.4.10_
, you may have accidentally left “t3.micro” in your settings; if so, go back to the Elastic Beanstalk settings and configure it to only use “t3.small” instance size.)
Install Success
Once the install succeeds, you’ll eventually see Platform Engine finished execution on command: app-deploy
.
data:image/s3,"s3://crabby-images/83579/83579b1a4498dfd669efd978c307855a2643faa1" alt="The CloudWatch 'Log groups' page, showing log output ending with 'Platform Engine finished execution'"
Back in CodePipeline, the Deploy stage will turn green for “Succeeded.”
data:image/s3,"s3://crabby-images/d5591/d559118659ad75570040914c1c9e15314b7506a2" alt="A CodePipeline page showing a Deploy step that has succeeded."
If you still have your browser tab open from the first time we checked the Elastic Beanstalk instance, you can reload it to see your running app. If not, go back to Elastic Beanstalk, pull up your environment, and click the “Domain” link again. You should see your “Rails on Elastic Beanstalk” message.
data:image/s3,"s3://crabby-images/b3ff8/b3ff84e465611285481e6e4bf1507d5319b42903" alt="A web page running on elasticbeanstalk.com. A heading says: 'Rails on Elastic Beanstalk'"
We’ve now got a Rails app running on Elastic Beanstalk, and each time you push commits up to the main
branch, they’ll be automatically deployed!
If that’s where you’d like to stop, we’ve made good progress. But if you have a custom domain name, we can add a custom subdomain and SSL to the app as well.
Adding a Custom Domain
AWS has its own service for registering custom domains and configuring DNS, called Route 53. However, you can use a different DNS provider, and I kind of like keeping my domains separate from any particular hosting platform. I use NameCheap.com for my DNS, so I’ll provide instructions for setting it up with NameCheap.
To make your Elastic Beanstalk instance accessible using a custom domain, we just need to create a CNAME record in your domain’s DNS. The “host” value should be the subdomain you want to use. In my case, I own codingitwrong.com
, so I’ll create a rails-eb
subdomain, so the site can be accessed at rails-eb.codingitwrong.com
. (This won’t be running by the time you read this post, so I don’t have to keep paying AWS for that server!)
For the CNAME “value,” put the domain name of your EB instance, without http://
on the front or a /
on the end, and with a .
added to the end. For example, my instance was http://rails-eb-env-1.eba-jad9rjd9.us-east-1.elasticbeanstalk.com/
, so for the “value” I put rails-eb-env-1.eba-jad9rjd9.us-east-1.elasticbeanstalk.com.
data:image/s3,"s3://crabby-images/ce5fa/ce5fa061ad3af9e94aad820fdccae8f0efcd93ca" alt="The NameCheap 'Advanced DNS' page for codingitwrong.com. A CNAME record for rails-eb has been added, pointing to a domain name on elasticbeanstalk.com"
After saving the DNS entry, depending on your DNS, it can take a little while to take effect. For me, it took 1-5 minutes. After that, I was able to go to http://rails-eb.codingitwrong.com
and see my running app. Nice!
data:image/s3,"s3://crabby-images/3a8de/3a8deb2be97fcd0db1494bc53e5f8ae5057262e0" alt="A browser showing domain rails-eb.codingitwrong.com. A heading says: 'Rails on Elastic Beanstalk'"
Adding HTTPS
A running app is good, but in 2024 you probably want HTTPS even if your site doesn’t handle any secure information. Thankfully, it’s not too hard to set this up with AWS: it will issue us an SSL certificate using the AWS Certificate Manager.
In the AWS search box, search for “Certificate Manager,” then open the “Certificate Manager” link in a new browser tab.
data:image/s3,"s3://crabby-images/baf2f/baf2f93ef789614a5e65605577e1d7a8b491ffe5" alt="The AWS Console search, with 'Certificate Manager' selected."
Click “Request a certificate.”
On the “Request certificate” screen, under “Certificate type,” make sure “Request a public certificate” is selected, then click “Next.”
On the “Request public certificate” screen, under “Domain names” > “Fully qualified domain name,” enter your full subdomain — in my case, rails-eb.codingitwrong.com
. Under “Validation method,” keep “DNS validation - recommended” selected. Click “Request.”
data:image/s3,"s3://crabby-images/d8888/d888859340da286419fe2c0021a1181ba51557ec" alt="The 'Request public certificate' screen of Certificate Manager. For the 'Fully qualified domain name,' 'rails-eb.codingitwrong.com' has been entered."
You will be taken to the Certificates page, and a message will show that says: “Successfully requested certificate with ID …. A certificate request with a status of pending validation has been created. Further action is needed to complete the validation and approval of the certificate.”
data:image/s3,"s3://crabby-images/c6590/c659091d176d3c6d9a73bc5c57383cd600bbf912" alt="The 'List certificates' page of Certificate Manager. A message says: 'Successfully requested certificate'"
Click the “View certificate” button.
You will see the page for your certificate. Under “Domains,” your domain will be listed with a status of “Pending validation.” You should see values in the “CNAME name” and “CNAME value” columns. If not, wait a few seconds and reload the page.
Once you have the CNAME name and value, you will need to create a CNAME entry under your domain to confirm you own it. Go back into your DNS provider and add them. Note that although the “CNAME name” has the domain suffix, in NameCheap at least I only needed to paste the “subdomain” part into the CNAME name field, so if AWS gave _123456890abcdef.rails-eb.codingitwrong.com.
, then I would only paste _123456890abcdef.rails-eb
. For the value, paste the full value ending in .acm-validations.aws.
Save the DNS record. Again, depending on how quickly your particular DNS propagates, it may take a few minutes before AWS Certificate Manager sees the result. For me, it took about five minutes. Keep reloading the certificate page, and when it’s working, you will see “Issued” for the status.
data:image/s3,"s3://crabby-images/7c1f2/7c1f2713327714bd62b8c4620092c6f11c54f3d6" alt="A page in Certificate Manager. The domain 'rails-eb.codingitwrong.com' is listed. The Status field says: 'Issued'"
After this, you’ll need to set up your Elastic Beanstalk instance to use that certificate. Go back to Elastic Beanstalk, open your environment, and go to “Configuration.” Under “Instance traffic and scaling,” click “Edit.”
data:image/s3,"s3://crabby-images/4d445/4d4450568752c2d24f3e35aba747ed16972a0780" alt="The Elastic Beanstalk Configuration page for environment 'Rails-eb-env-1'. The 'Instance traffic and scaling' box is shown."
Under “Capacity” > “Auto scaling group” > “Environment type,” change it from “Single instance” to “Load balanced.” This isn’t because we need more than one instance; we just need a load balancer to set up HTTPS on. Under “Instances,” change “Max” from 4 to 1 to avoid being charged for multiple instances.
data:image/s3,"s3://crabby-images/9ff89/9ff891fa09d36f6e5bd3a4207e71e516109132e8" alt="The 'Capacity' box in Elastic Beanstalk. 'Environment type' is set to 'Load balanced', and 'Instances' has a minimum of 1 and a maximum of 1."
Now that you’ve changed the environment type to load balanced, further down on the page, you should see a “Listeners” section with port 80 listed.
data:image/s3,"s3://crabby-images/26d20/26d20750469364481c5e761866274d7f3a2074dd" alt="The Listeners box in Elastic Beanstalk. Only listener port 80 is configured."
Click “Add listener.” For “Listener port,” type 443, the standard HTTPS port number. For “Listener protocol,” choose HTTPS. For “SSL certificate,” choose the certificate for the subdomain that you just created.
data:image/s3,"s3://crabby-images/a274c/a274cddca470c3bdf1b4a53d18ba0abc258d7d0a" alt="The 'Add listener' dialog in Elastic Beanstalk. It is configured with Listener Port 443, Listener Protocol HTTPS, and SSL Certificate 'rails-eb.codingitwrong.com'"
Then click “Save.” There’s one more step: scroll down to the bottom of the page, then click “Apply.”
data:image/s3,"s3://crabby-images/15268/15268533fb3f730499910f25f0a22c7d7f3a8a2e" alt="The Configure Instance Traffic screen of Elastic Beanstalk. The Apply button is visible."
Elastic Beanstalk will take some time to set up the new load balancer. Watch the Events and wait for it to say “Environment update successfully completed.”
data:image/s3,"s3://crabby-images/e9cd9/e9cd91d1e689a87ef187af4c9e80810091f5c528" alt="The Elastic Beanstalk Events tab for environment 'Rails-eb-env-1'. A message says: 'Environment update successfully completed."
Now, go to the https://
version of your URL, which, in my case, is https://rails-eb.codingitwrong.com
. It may still take a few seconds before it’s ready (it did in my case), but soon, your app should be available over HTTPS!
data:image/s3,"s3://crabby-images/7b5a2/7b5a22a79b3fcc865db9ccbeddd0647545894483" alt="A web browser showing the site https://rails-eb.codingitwrong.com. The security menu says: 'You are securely connected to this site.'"
Now that that’s working, it would be best for us to re-enable Rails force_ssl
, so that users can’t accidentally access it over HTTP. Change the config value back:
# Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
-config.force_ssl = false # TEMPORARY for testing
+config.force_ssl = true
You can change your welcome message, too, so you can be sure when the update is deployed:
-<h1>Rails on Elastic Beanstalk</h1>
+<h1>SECURE Rails on Elastic Beanstalk</h1>
Commit these changes and push them up to GitHub. You should see a redeployment be triggered in the Elastic Beanstalk Events.
Once the deployment is done, it’s a good idea to test that SSL protection is in place. Run the following from the terminal, putting in your subdomain (note the HTTP instead of HTTPS):
$ curl -v http://rails-eb.codingitwrong.com
If this outputs the HTML of your web page, this means the SSL protection is not working. But if it is working, you will see something like the following:
< HTTP/1.1 301 Moved Permanently
< Date: Sat, 17 Feb 2024 11:17:38 GMT
< Content-Type: text/html; charset=utf-8
< Content-Length: 0
< Connection: keep-alive
< Server: nginx
< location: https://rails-eb.codingitwrong.com/
The 301 redirecting the user to the https
URL means that Rails is not serving up the app over HTTP. Nice and secure!
Now you’re all set with a Rails app running on Elastic Beanstalk with a custom domain and HTTPS. Remember that you’ll be charged for these running services, so be sure to spin them down if you want to avoid that. You can always run through this tutorial again in the future to get another Rails app set up.
Resources
If you’d like to learn more about AWS, here are some of the resources I used while getting my Rails app running:
- Deploying a Rails application to Elastic Beanstalk by AWS
- AWS Elastic Beanstalk gotchas with Rails by Rachel Chervin
- How to map custom domain name for AWS Elastic Beanstalk by Colin Toh
- Udemy course: Amazon Web Services (AWS) with JavaScript & NodeJS by Parwiz Forogh