-
Notifications
You must be signed in to change notification settings - Fork 0
Deployment guide
The deployment system is implemented with Vlad. The script is in config/deploy.rb
. This file contains server settings, deployment path and the git repository URL at the start.
The script currently uses Ruby through rbenv, which allows installation of several Ruby versions and easy switching between them. The different versions are under the directory ~/.rbenv/versions
on the server and the version the application uses is specified in the file .rbenv-version
in the git repository. Please note that Phusion Passenger doesn't support rbenv and if the application's Ruby version is changed, Passenger's configuration (in /opt/nginx/conf/nginx.conf) must be changed to reflect the change.
When everything is set up, the deployment is done with the following command:
rake vlad:deploy
You need a private ssh key to access the Amazon EC2 server (e.g. in ~/.ssh/aapps).
Ssh in the local computer needs to be set up to use the correct ssh key to when connecting to EC2 server. This can be done by adding the following lines to ~/.ssh/config
:
Host aaltoapps.fi aaltoapps.com aaltoapps.net
IdentityFile ~/.ssh/aapps
User aaltoapps
ForwardAgent yes
This guide is for Ubuntu 11.10 running in an Amazon EC2 server.
-
Create a new user account. The application will run under this account and the account will be used for deployment. The following steps are done as this user. The user needs sudo privileges for this to work.
-
Install the following packages (
sudo apt-get install package1 package2 ...
):- gcc
- g++
- make
- automake1.9
- git
- postgresql
- nodejs or some other Javascript runtime that works with ExecJS
- imagemagick
- it's probably best to install this with the apt-get flag
--no-install-recommends
- it's probably best to install this with the apt-get flag
- zlib1g-dev
- libcurl4-openssl-dev or libcurl4-gnutls-dev
- libssl-dev
- libpq-dev
- language-pack-en and other language packs you might use
- PostgreSQL uses these for collation and other locale-specific string handling
- install these with the apt-get flag
--no-install-recommends
to prevent the installation of Firefox language packs
If some of the -dev-packages are not installed, ruby-build may silently skip building some of its core modules, which some gems may require.
-
Install ruby-build:
git clone git://github.com/sstephenson/ruby-build.git cd ruby-build sudo ./install.sh
-
Install rbenv:
cd git clone git://github.com/sstephenson/rbenv.git .rbenv echo 'export PATH="$HOME/.rbenv/bin:$PATH"' >> .profile echo 'eval "$(rbenv init -)"' >> .profile
-
Install
the latest stableRuby under~/.rbenv/versions/
and set it as the default ruby:ruby-build 1.9.2-p290 $HOME/.rbenv/versions/1.9.2-p290 rbenv rehash rbenv global 1.9.2-p290
Note: As of version 3.0.11 Phusion Passenger doesn't seem to work with Ruby 1.9.3.
-
Install Nginx and Phusion Passenger:
gem install passenger --no-rdoc --no-ri rbenv rehash sudo sh -c "PATH=\"$PATH\" passenger-install-nginx-module \ --auto \ --prefix=/opt/nginx \ --auto-download \ --extra-configure-flags='\ --with-http_gzip_static_module \ --with-http_ssl_module'"
The option
--with-http_ssl_module
is optional.-
Create a directory for the application and give yourself write permissions to it:
sudo mkdir -p /deployment_path sudo chown your_username:your_groupname /deployment_path
In Ubuntu your_groupname is typically the same as your_username.
-
Edit Nginx configuration in file
/opt/nginx/conf/nginx.conf
. For basic installation replaceserver
section with the following:server { listen 80; server_name domain.name.of.the.server alternate.domain.name ...; root /deployment_path/current/public; passenger_enabled on; location ~ ^/assets/ { # Some browsers still send conditional-GET requests if there's a # Last-Modified header even if they haven't reached the expiry # date sent in the Expires header. The same applies to ETag # headers, but Nginx does't generate them. add_header Last-Modified ""; gzip_static on; # serve pre-gzipped versions of files expires max; add_header Cache-Control public; } }
-
passenger-install-nginx-module
doesn't seem to create init script for Nginx, so we have to do it ourselves:-
Create file
/etc/init.d/nginx
as root with the following contents:#!/bin/sh ### BEGIN INIT INFO # Provides: nginx # Required-Start: $all # Required-Stop: $all # Default-Start: 2 3 4 5 # Default-Stop: 0 1 6 # Short-Description: starts the nginx web server # Description: starts nginx using start-stop-daemon ### END INIT INFO PATH=/opt/nginx/sbin:/sbin:/bin:/usr/sbin:/usr/bin DAEMON=/opt/nginx/sbin/nginx NAME=nginx DESC=nginx test -x $DAEMON || exit 0 # Include nginx defaults if available if [ -f /etc/default/nginx ] ; then . /etc/default/nginx fi set -e case "$1" in start) echo -n "Starting $DESC: " start-stop-daemon --start --quiet --pidfile /opt/nginx/logs/$NAME.pid \ --exec $DAEMON -- $DAEMON_OPTS echo "$NAME." ;; stop) echo -n "Stopping $DESC: " start-stop-daemon --stop --quiet --pidfile /opt/nginx/logs/$NAME.pid \ --exec $DAEMON echo "$NAME." ;; restart|force-reload) echo -n "Restarting $DESC: " start-stop-daemon --stop --quiet --pidfile \ /opt/nginx/logs/$NAME.pid --exec $DAEMON sleep 1 start-stop-daemon --start --quiet --pidfile \ /opt/nginx/logs/$NAME.pid --exec $DAEMON -- $DAEMON_OPTS echo "$NAME." ;; reload) echo -n "Reloading $DESC configuration: " start-stop-daemon --stop --signal HUP --quiet --pidfile /opt/nginx/logs/$NAME.pid \ --exec $DAEMON echo "$NAME." ;; *) N=/etc/init.d/$NAME echo "Usage: $N {start|stop|restart|reload|force-reload}" >&2 exit 1 ;; esac exit 0
(copied from http://library.linode.com/frameworks/ruby-on-rails-nginx/ubuntu-10.10-maverick)
-
sudo chmod +x /etc/init.d/nginx
-
Run the command:
sudo update-rc.d -f nginx defaults
-
Start Nginx:
sudo /etc/init.d/nginx start
-
-
-
Set up PostgreSQL database
-
Edit
/etc/postgresql/9.1/main/pg_hba.conf
as root and add the line:local all all md5
-
Restart PostgreSQL (
sudo /etc/init.d/postgresql restart
) -
sudo -u postgres psql postgres
:postgres=# CREATE ROLE aaltoapps WITH LOGIN PASSWORD 'database_password'; postgres=# CREATE DATABASE aaltoapps OWNER aaltoapps; postgres=# \q
-
-
Set up mail server
The application needs a mail server to send notification emails. The sending of these emails is a blocking operation, which means that when a user performs an action that causes an email to be sent, the rendering of the page is delayed until the email has been sent or an error or a timeout occurs. Therefore it is recommended that a local email server is installed even if it just sends all the emails to another mail server.
To install Postfix and configure it to only send mail run the following command:
sudo apt-get install postfix
Select "Satellite system". If the emails are to be sent directly to the recipient's server, set "SMTP relay host" to blank. Double-check that the mail server only listens for connections from the loopback interface.
-
Configure and deploy the application
-
do the following on your computer (not on the server):
- Set up a development environment
- Make sure that settings in the file
config/deploy.rb
are correct and that you have ssh access to the server - Run the following command in the development directory:
bundle exec rake vlad:setup
-
do the following on the server:
-
run these commands:
cd /deployment_path mkdir shared/config touch shared/config/{aaltoapps_config.yml,database.yml} chmod go= shared/config/{aaltoapps_config.yml,database.yml}
-
edit
shared/config/database.yml
:production: username: aaltoapps password: database_password database: aaltoapps adapter: postgresql
-
Copy config/aaltoapps_config.yml or config/aaltoapps_config_example.yml in your development directory to
shared/config/aaltoapps_config.yml
and edit it to suit your needs -
run the command:
gem install bundler --no-rdoc --no-ri
-
-
on your computer in the development directory:
bundle exec rake vlad:deploy
If the deployment fails, it may be because of a missing library. In that case install the library on the server and try again.
-