123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- #
- # Cookbook Name:: application
- # Recipe:: django
- #
- # Copyright 2011, Opscode, Inc.
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- #
- app = node.run_state['current_app']
- include_recipe "python"
- ###
- # You really most likely don't want to run this recipe from here - let the
- # default application recipe work it's mojo for you.
- ###
- node.default['apps'][app['id']][node.chef_environment]['run_migrations'] = false
- # the Django split-settings file name varies from project to project...+1 for standardization
- local_settings_full_path = app['local_settings_file'] || 'settings_local.py'
- local_settings_file_name = local_settings_full_path.split(/[\\\/]/).last
- ## Create required directories
- directory app['deploy_to'] do
- owner app['owner']
- group app['group']
- mode '0755'
- recursive true
- end
- directory "#{app['deploy_to']}/shared" do
- owner app['owner']
- group app['group']
- mode '0755'
- recursive true
- end
- ## Create a virtualenv for the app
- ve = python_virtualenv app['id'] do
- path "#{app['deploy_to']}/shared/env"
- action :create
- end
- ## First, install any application specific packages
- if app['packages']
- app['packages'].each do |pkg,ver|
- package pkg do
- action :install
- version ver if ver && ver.length > 0
- end
- end
- end
- ## Next, install any application specific gems
- if app['pips']
- app['pips'].each do |pip,ver|
- python_pip pip do
- version ver if ver && ver.length > 0
- virtualenv ve.path
- action :install
- end
- end
- end
- if app.has_key?("deploy_key")
- ruby_block "write_key" do
- block do
- f = ::File.open("#{app['deploy_to']}/id_deploy", "w")
- f.print(app["deploy_key"])
- f.close
- end
- not_if do ::File.exists?("#{app['deploy_to']}/id_deploy"); end
- end
- file "#{app['deploy_to']}/id_deploy" do
- owner app['owner']
- group app['group']
- mode '0600'
- end
- template "#{app['deploy_to']}/deploy-ssh-wrapper" do
- source "deploy-ssh-wrapper.erb"
- owner app['owner']
- group app['group']
- mode "0755"
- variables app.to_hash
- end
- end
- if app["database_master_role"]
- dbm = nil
- # If we are the database master
- if node.run_list.roles.include?(app["database_master_role"][0])
- dbm = node
- else
- if Chef::Config[:solo]
- Chef::Log.warn("This recipe uses search. Chef Solo does not support search.")
- else
- # Find the database master
- results = search(:node, "role:#{app["database_master_role"][0]} AND chef_environment:#{node.chef_environment}", nil, 0, 1)
- rows = results[0]
- if rows.length == 1
- dbm = rows[0]
- end
- end
- end
-
- # we need the django version to render the correct type of settings.py file
- django_version = 1.2
- if app['pips'].has_key?('django') && !app['pips']['django'].strip.empty?
- django_version = app['pips']['django'].to_f
- end
- # Assuming we have one...
- if dbm
- # local_settings.py
- template "#{app['deploy_to']}/shared/#{local_settings_file_name}" do
- source "settings.py.erb"
- owner app["owner"]
- group app["group"]
- mode "644"
- variables(
- :host => (dbm.attribute?('cloud') ? dbm['cloud']['local_ipv4'] : dbm['ipaddress']),
- :database => app['databases'][node.chef_environment],
- :django_version => django_version
- )
- end
- else
- Chef::Log.warn("No node with role #{app["database_master_role"][0]}, #{local_settings_file_name} not rendered!")
- end
- end
- ## Then, deploy
- deploy_revision app['id'] do
- revision app['revision'][node.chef_environment]
- repository app['repository']
- user app['owner']
- group app['group']
- deploy_to app['deploy_to']
- action app['force'][node.chef_environment] ? :force_deploy : :deploy
- ssh_wrapper "#{app['deploy_to']}/deploy-ssh-wrapper" if app['deploy_key']
- shallow_clone true
- purge_before_symlink([])
- create_dirs_before_symlink([])
- symlinks({})
- before_migrate do
- requirements_file = nil
- # look for requirements.txt files in common locations
- if ::File.exists?(::File.join(release_path, "requirements", "#{node['chef_environment']}.txt"))
- requirements_file = ::File.join(release_path, "requirements", "#{node.chef_environment}.txt")
- elsif ::File.exists?(::File.join(release_path, "requirements.txt"))
- requirements_file = ::File.join(release_path, "requirements.txt")
- end
-
- if requirements_file
- Chef::Log.info("Installing pips using requirements file: #{requirements_file}")
- pip_cmd = File.join(ve.path, "bin", "pip")
- execute "#{pip_cmd} install -r #{requirements_file}" do
- ignore_failure true
- cwd release_path
- end
- end
- end
- symlink_before_migrate({
- local_settings_file_name => local_settings_full_path
- })
- if app['migrate'][node.chef_environment] && node['apps'][app['id']][node.chef_environment]['run_migrations']
- migrate true
- migration_command app['migration_command'] || "#{::File.join(ve.path, "bin", "python")} manage.py migrate"
- else
- migrate false
- end
- before_symlink do
- ruby_block "remove_run_migrations" do
- block do
- if node.role?("#{app['id']}_run_migrations")
- Chef::Log.info("Migrations were run, removing role[#{app['id']}_run_migrations]")
- node.run_list.remove("role[#{app['id']}_run_migrations]")
- end
- end
- end
- end
- end
|