Skip to content

Instantly share code, notes, and snippets.

@bennylope
Created April 30, 2014 00:35
Show Gist options
  • Save bennylope/23e53ac31808b3affecf to your computer and use it in GitHub Desktop.
Save bennylope/23e53ac31808b3affecf to your computer and use it in GitHub Desktop.
Deployment w/ Capistrano style deployments with Ansible & Capistrano
---
- name: Deploy new site release
user: deployer
hosts: all
tasks:
- name: Fetch repo updates
git: >
[email protected]:my/repo.git
dest="{{ git_cache_path }}"
version=master
accept_hostkey=yes
- name: Get release timestamp
command: date +%Y%m%d%H%M%S
register: timestamp
- name: Name release directory
command: echo "{{ releases_dir }}/{{ timestamp.stdout }}"
register: release_path
- name: Create release directory
file: >
state=directory
owner=deployer
group=wheel
recurse=yes
path={{ release_path.stdout }}
- name: Cop release
command: rsync -avz -avz --exclude-from '{{ git_cache_path }}/.slugignore' {{ git_cache_path }}/ {{ release_path.stdout }}
- name: Update app version
file: >
state=link
path=/var/apps/cis_env/cis
src={{ release_path.stdout }}
- name: Link uploads folder
file:
state=link
path="{{ release_path.stdout }}/app/assets/uploads"
src={{ uploads_dir }}
load 'deploy' if respond_to?(:namespace) # cap2 differentiator
# general config
set :application, "myapp"
set :valid_environments, ["staging","production"]
set :user, "deployer"
# git config
set :scm, :git
set :repository, "[email protected]:my/repo.git"
set :branch, "master"
set :deploy_via, :remote_cache
default_run_options[:pty] = true
# set :use_sudo, true
# ssh_options[:forward_agent] = true
set :current_dir, "myapp"
namespace :deploy do
desc "Sets up basic directory structure on target server"
task :setup, :except => { :no_release => true } do
dirs = [deploy_to, releases_path, shared_path]
dirs += shared_children.map { |d| File.join(shared_path, d) }
run "#{try_sudo} mkdir -p #{dirs.join(' ')} && #{try_sudo} chmod g+w #{dirs.join(' ')}"
end
desc <<-DESC
Updates the symlink to the most recently deployed version. Capistrano works \
by putting each new release of your application in its own directory. When \
you deploy a new version, this task's job is to update the `current' symlink \
to point at the new version. You will rarely need to call this task \
directly; instead, use the `deploy' task (which performs a complete \
deploy, including `restart') or the 'update' task (which does everything \
except `restart').
DESC
task :create_symlink, :except => { :no_release => true } do
on_rollback do
if previous_release
run "rm -f #{current_path}; ln -s #{previous_release} #{current_path}; true"
else
logger.important "no previous release to rollback to, rollback of symlink skipped"
end
end
# delete the old and create a new symlink to the latest release
run "rm -f #{current_path} && ln -s #{latest_release} #{current_path}"
# delete the uploads directory out of the new codebase if it exists. then symlink the uploads directory to there.
# run "rm -rf #{latest_release}/app/assets/uploads && ln -s #{deploy_to}/shared/uploads #{latest_release}/app/assets/uploads"
run "ln -s #{deploy_to}/shared/uploads #{latest_release}/app/assets/uploads"
end
desc <<-DESC
Restarts your application.
By default, this will be invoked via sudo as the 'app' user. If
you are in an environment where you can't use sudo, set
the :use_sudo variable to false:
set :use_sudo, false
DESC
task :restart, :roles => :app, :except => { :no_release => true } do
run "#{sudo} chown -R www-data:www-data /var/apps/myapp/shared/uploads/"
run "#{sudo} /etc/init.d/apache2 reload"
end
desc <<-DESC
[internal] Touches up the released code. This is called by update_code \
after the basic deploy finishes.
This task will make the release group-writable (if the :group_writable \
variable is set to true, which is the default). It will then change \
ownership of the release and the symlink that was created if the owners \
variable exists.
DESC
task :finalize_update, :except => { :no_release => true } do
# run "chmod -R g+w #{latest_release}" if fetch(:group_writable, true)
run "#{sudo} chown -R #{owners} #{current_path}" if exists?("owners")
run "#{sudo} chown -R #{owners} #{latest_release}" if exists?("owners")
end
end

Project: Django site for which original deployment was scripted with Capistrano. Updating the site and using Ansible to wrangle the server configurations. Why not deploy with it too?

Observed that for similar "Capistrano style" deployments, Capistrano is typically much faster at executing a deployment.

I've noticed a similar discrepancy between Fabric and Capistrano and suspect it comes down to the model of executing discrete commands (Ansible and Fabric) vs. bundling shell commands (Capistrano). This is based on observed output of Capistrano and my understanding of how Ansible and Fabric execute commands, which I do not assert to be authoritative!

Using Capistrano v2.15.4 and Ansible 1.5.4. Ran from my computer over my home network with Verizon FiOS. Target server is a small testing server on Rackspace:

Ansible total time: 1:50.44 total (43.821 total from a Digital Ocean instance) Capistrano total time: 16.951 total

The Ansible deployment playbook was written to try to follow the Capistrano steps as closely as possible (difference in using rsync vs. Capistrano's cp, however).

@mpdehaan
Copy link

Various steps above in ansible could be done in a preceding play and set using "local_action" versus remote ops. Thus the two steps you have above are not quite equivalent.

Also make sure you are using Control Persist and pipelining mode for maximum performance benefits, (see ansible.cfg for pipelining, off by default).

Ultimately though the amount of time you have posted above makes it sound like there's really a DNS issue going on here.

If you are in fact using -c ssh on a system that isn't new enough to support ControlPersist, you're better off with ansible using paramiko, which is what fabric uses. Depends on what OS you are running but Ansible will choose a default smartly.

I'm also a bit confused by the git checkout and then doing the rsync, where as I believe Fabric is just doing a symlink. Still, rsync should be fast locally, I'm not quite following the need for that step if everything is in git and ansible is also creating the symlink.

I'd probably also check to see if you are benchmarking repeated runs and such, but I'm unable to read the Fabric enough to understand what it's doing.

@mpdehaan
Copy link

Oh the above is Capistrano, sorry for quick skimming :) ... explains why the Fabric looked like Ruby :)

I'd definitely check what connection type you are using in Ansible and how it's tuned in your case.

Could also be a DNS issue -- maybe.

Still think the extra rsync is involved.

@bennylope
Copy link
Author

This is incredibly helpful.

I'm also a bit confused by the git checkout and then doing the rsync, where as I believe Fabric is just doing a symlink. Still, rsync should be fast locally, I'm not quite following the need for that step if everything is in git and ansible is also creating the symlink.

That's making a discrete release that's distinct from the cached copy of the repo. It means you can execute commands against a single release or rollback without fussing with the history of your local Git cache.

@mpdehaan
Copy link

I would be to know whether capistrano is checking out the whole history or just the head revision?

There's a bit too much automagic here to see where it's checking out the repo and doing the git commands.

SSH overhead between tasks, even untuned and using the wrong connection plugins, should be around .5 seconds or less, and you have a small number of tasks here -- so something is definitely up and/or very different.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment