Goal: Understand the steps needed to create, version, and deploy a SaaS app, including tracking the libraries it depends on so that your production and development environments are as similar as possible.
What you will do: Create a simple "hello world" app using the Sinatra framework, version it properly, and deploy it to Heroku.
SaaS apps are developed on your computer (or cloud-based IDE) but deployed to production on a server that others can access. We try to minimize the differences between the development and production environments, to avoid difficult-to-diagnose problems in which something works one way on your development computer but a different way (or not at all) when that code is deployed to production.
We have two mechanisms for keeping the development and production environments consistent. The first is version control, such as Git, for the app's code. But since almost all apps also rely on libraries written by others, such as gems in the case of Ruby, we need a way to keep track of which versions of which libraries our app has been tested with, so that the same ones are used in development and production.
Happily, Ruby has a wonderful system for managing gem dependencies: a gem called Bundler looks for a file called Gemfile
in the app root directory of each project. The Gemfile
contains a list of gems and versions your app depends on. Bundler verifies that those gems, and any others that they in turn depend on, are properly installed on your system and accessible to the app.
Let's start with the following steps:
-
Create a new empty directory to hold your new app, and use
git init
in that directory to start versioning it with Git. -
In that directory, create a new file called
Gemfile
(the capitalization is important) with the following contents. This file will be a permanent part of your app and will travel with your app anywhere it goes:
source 'https://rubygems.org'
ruby '2.4.0'
gem 'sinatra', '>= 2.0.1'
The first line says that the preferred place to download any necessary gems is https://rubygems.org, which is where the Ruby community registers "production ready" gems.
The second line specifies which version of the Ruby language interpreter is required. If we omitted this line, Bundler wouldn't try to verify which version of Ruby is available; there are subtle differences between the versions, and not all gems work with all versions, so it's best to specify this.
The last line says we need version 2.0.1 or later of the sinatra
gem. In some cases we don't need to specify which version of a gem we want; in this case we do specify it because we rely on some features that are absent from earlier versions of Sinatra.
Run the command bundle
, which examines your Gemfile
to make sure the correct gems (and, where specified, the correct versions) are available, and tries to install them otherwise. This will create a new file Gemfile.lock
, which you should place under version control.
To place under version control, use these commands:
$ git add .
$ git commit -m "Set up the Gemfile"
The first command stages all changed files for committing. The second command commits the staged files with the comment in the quotes. You can repeat these commands to commit future changes. Remember that these are LOCAL commits -- if you want these changes on GitHub, you'll need to do a git push command, which we will show later.
What's the difference between the purpose and contents of Gemfile
and Gemfile.lock
? Which file is needed to completely reproduce the development environment's gems in the production environment?
Gemfile
specifies the gems you need and in some cases the constraints on which version(s) are acceptable.Gemfile.lock
records the *actual* versions found, not only of the gems you specified explicitly but also any other gems on which they depend, so it is the file used by the production environment to reproduce the gems available in the development environment.
After running bundle
, why are there gems listed in Gemfile.lock
that were not listed in Gemfile
?
Bundler looked up the information for each Gem you requested (in this case, onlysinatra
) and realized that it depends on other gems, which in turn depend on still others, so it recursively installed all those dependencies. For example, therack
appserver is a gem, and while you didn't explicitly request it,sinatra
depends on it. This is an example of the power of automation: rather than requiring you (the app developer) to understand every Gem dependency, Bundler automates that process and lets you focus only on your app's top-level dependencies.
As Chapter 2 of ESaaS explains, SaaS apps require a web server to receive HTTP requests from the outside world, and an application server that "connects" your app's logic to the web server. For development, we will use webrick
, a very simple Ruby-based web server that would be inappropriate for production but is fine for development. In both development and production, we will use the rack
Ruby-based application server, which supports Ruby apps written in various frameworks including Sinatra and Rails.
As Chapter 2 of ESaaS explains, a SaaS app essentially recognizes and responds to HTTP requests corresponding to the application's routes (recall that a route consists of an HTTP method such as GET
or POST
plus a URI). Sinatra provides an extremely lightweight shorthand for matching a route with the app code to be executed when a request using that route arrives from the Web server.
Create a file in your project called app.rb
containing the following:
require 'sinatra'
class MyApp < Sinatra::Base
get '/' do
"<!DOCTYPE html><html><head></head><body><h1>Hello World</h1></body></html>"
end
end
The get
method is provided by the Sinatra::Base
class, from which our MyApp
class inherits; Sinatra::Base
is available because we load the Sinatra library on line 1.
What *two* steps did we take earlier to guarantee that the Sinatra library is available to load in line 1?
We specifiedgem 'sinatra'
in theGemfile
*and* successfully ranbundle
to confirm that the gem is installed and "lock" the correct version of it inGemfile.lock
.
As you see from the above simple example, Sinatra lets you write functions that match an incoming HTTP route, in this case GET '/'
(the root URL), a very simple HTML document containing the string Hello World
will be returned to the presentation tier as the result of the request.
To run our app, we have to start the application server and presentation tier (web) server. The rack
application server is controlled by a file config.ru
, which you must now create and add to version control, containing the following:
require './app'
run MyApp
The first line tells Rack that our app lives in the file app.rb
, which you created above to hold your app's code. We have to explicitly state that our app
file is located in the current directory (.) because require
normally looks only in standard system directories to find gems.
If you're using Cloud9, you're now ready to test-drive our simple app with this command line:
$ bundle exec rackup -p $PORT -o $IP
Available ports on a hosted Cloud9 workspace
This command starts the Rack appserver and the WEBrick webserver. Prefixing it with bundle exec
ensures that you are running with the gems specified in Gemfile.lock
. Rack will look for config.ru
and attempt to start our app based on the information there. If you're using Cloud9, you will see a small popup in the terminal with a URL to your running webapp. It will open in a new tab in the IDE if you click on it, but you should open up a fresh browser tab and paste in that URL.
Point a new Web browser tab at the running app's URL and verify that you can see "Hello World".
What happens if you try to visit a non-root URL such as https://workspace-username.c9.io/hello
and why? (your URL root will vary)
You'll get a humorous error message from the Sinatra framework, since you don't have a route matching get '/hello'
in your app. Since Sinatra is a SaaS framework, the error message is packaged up in a Web page and delivered to your browser.
You should now have the following files under version control: Gemfile
, Gemfile.lock
, app.rb
, config.ru
. This is a minimal SaaS app: the app file itself, the list of explicitly required gems, the list of actual gems installed including the dependencies implied by the required gems, and a configuration file telling the appserver how to start the app.
Modify app.rb
so that instead of "Hello World" it prints "Goodbye World". Save your changes to app.rb
and try refreshing your browser tab where the app is running.
No changes? Confused?
Now go back to the shell window where you ran rackup
and press Ctrl-C to stop Rack. Then type bundle exec rackup -p $PORT -o $IP
again (for Cloud9), and once it is running, go back to your browser tab with your app and refresh the page. This time it should work.
What this shows you is that if you modify your app while it's running, you have to restart Rack in order for it to "see" those changes. Since restarting it manually is tedious, we'll use the rerun
gem, which restarts Rack automatically when it sees changes to files in the app's directory. (Rails does this for you by default during development, as we'll see, but Sinatra doesn't.)
You're probably already thinking: "Aha! If our app depends on this additional gem, we should add it to the Gemfile and run bundle to make sure it's really present." Good thinking. But it may also occur to you that this particular gem wouldn't be necessary in a production environment: we only need it as a tool while developing. Fortunately, there's a way to tell Bundler that some gems are only necessary in certain environments. Add the following to the Gemfile (it doesn't matter where):
group :development do
gem 'rerun'
end
Any gem specifications inside the group :development
block will only be examined if bundle is run in the development environment. (The other environments you can specify are :test and :production, and you can define new environments yourself.) Gem specifications outside of any group block are assumed to apply in all environments.
Say bundle exec rerun -- rackup -p $PORT -o $IP
in the terminal window to start your app and verify the app is running. There are more details on rerun's usage available in the gem's GitHub README. Gem's are usually on GitHub and their README's full of helpful instructions about how to use them.
In this case we are prefixing with bundle exec
again in order to ensure we are using the gems in the Gemfile.lock, and the --
symbol is there to assert that the command we want rerun to operate with is rackup -p $PORT -o $IP
. We could achieve the same effect with bundle exec rerun "rackup -p $PORT -o $IP"
. They are equivalent. More importantly any detected changes will now cause the server to restart automatically, similar to the use of guard
to auto re-run specs when files change.
Modify app.rb
to print a different message, and verify that the change is detected by rerun byagain refreshing your browser tab with the running app. Also before we move on you should commit your latest changes to git.
Heroku is a cloud platform-as-a-service (PaaS) where we can deploy our Sinatra (and later Rails) applications in a more robust way than via Cloud9. If you don't have an account yet, go sign up at http://www.heroku.com. You'll need your login and password for the next step.
If using Cloud9, update your Heroku Toolbelt installation by typing the following command:
$ wget -O- https://toolbelt.heroku.com/install-ubuntu.sh | sh
Log in to your Heroku account by typing the command: heroku login
in the Cloud9 terminal. This will connect your Cloud9 workspace to your Heroku account.
While in the root directory of your project (not your whole workspace), type heroku create
to create a new project in Heroku. This will tell the Heroku service to prepare for some incoming code, and locally on Cloud9, it will add a remote git repository for you called heroku
.
Next, make sure you stage and commit all changes locally as instructed above (i.e. git add
, git commit
, etc).
Earlier we saw that to run the app locally you run rackup
to start the Rack appserver, and Rack looks in config.ru
to determine how to start your Sinatra app. How do you tell a production environment how to start an appserver or other processes necessary to receive requests and start your app? In the case of Heroku, this is done with a special file named Procfile
, which specifies one or more types of Heroku processes your app will use, and how to start each one. The most basic Heroku process type is called a Dyno, or "web worker". One Dyno can serve one user request at a time. Since we're on Heroku's free tier, we can only have one Dyno. Let's create a file named Procfile
, and only this as the name (i.e. Procfile.txt is not valid). Write the following line in your Procfile
:
web: bundle exec rackup config.ru -p $PORT
This tells Heroku to start a single web worker (Dyno) using essentially the same command line you used to start Rack locally. Note that in some cases, a Procfile
is not necessary since Heroku can infer from your files how to start the app. However, it's always better to be explicit.
Your local Cloud9 repo is now ready to deploy to Heroku:
$ git push heroku master
(master
refers to which branch of the remote Heroku repo we are pushing to. We'll learn about branches later in the course, but for now, suffice it to say that you can only deploy to the master
branch on Heroku.) This push will create a running instance of your app at some URL ending with herokuapp.com
. Enter that URL in a new browser tab (not in the Cloud9 IDE) to see your app running live. Congratulations, you did it--your app is live!
-
You started a new application project by creating a
Gemfile
specifying which gems you need and runningbundle
to verify that they're available and create theGemfile.lock
file that records the versions of gems actually in use. -
You created a Sinatra app in the file
app.rb
, pointed Rack at this file inconfig.ru
, and usedrackup
to start the appserver and the WEBrick web server. -
You learned that changing the app's code doesn't automatically cause Rack to reload the app. To save the work of restarting the app manually every time you make a change, you used the
rerun
gem, adding it to the Gemfile in a way that specifies you won't need it in production, only during development. -
You versioned the important files containing not only your app's code but the necessary info to reproduce all the libraries it relies on and the file that starts up the app.
-
You deployed this simple app to Heroku.
Next: Part 1 - Hangperson