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. 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.
-
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.3.0'
gem 'sinatra', '>= 1.4'
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 1.4 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 -A
$ git commit -am "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, only `sinatra`) and realized that it depends on other gems, which in turn depend on still others, so it recursively installed all those dependencies. For example, the `rack` 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 in line 1.
What *two* steps did we take earlier to guarantee that the Sinatra library is available to load in line 1?
We specified `gem 'sinatra'` in the `Gemfile` *and* successfully ran `bundle` to confirm that the gem is installed and "lock" the correct version of it in `Gemfile.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 version, 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:
$ rackup -p $PORT -o $IP
This command starts the Rack appserver and the WEBrick webserver. 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 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 rerun -- rackup -p $PORT -o $IP
in the terminal window to start
your app and verify the app is running. Note that you need to put
everything except rerun
in quotes in order to make the command
unambiguous in Cloud9. Now any detected changes will cause the server to
restart automatically, similar to the use of guard
to run specs.
Modify app.rb
to print a different message, and verify that the change is detected by rerun by again refreshing your browser tab with the running app.
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