1 from __future__ import with_statement
 
   8 logging.basicConfig(stream=sys.stderr, format="%(levelname)s:: %(message)s", level=logging.INFO)
 
  10 from string import Template
 
  12 class DeploySite(object):
 
  14     def __init__(self, **env):
 
  17         for arg in ('ROOT', 'PROJECT_NAME', 'PYTHON_VERSION'):
 
  18             if arg not in self.env:
 
  19                 raise ValueError("Argument '%s' is required." % arg)
 
  21         if 'PYTHON_BASE' not in self.env:
 
  22             self.env['PYTHON_BASE'] = os.path.join(self.env['ROOT'], 'pythonenv')
 
  24         if 'PYTHON_BIN' not in self.env:
 
  25             self.env['PYTHON_BIN'] = os.path.join(
 
  26                         self.env['PYTHON_BASE'], 'bin', 'python') + self.env['PYTHON_VERSION']
 
  28         if 'PIP_BIN' not in self.env:
 
  29             self.env['PIP_BIN'] = os.path.join(self.env['PYTHON_BASE'], 'bin', 'pip')
 
  31         if 'PYTHON_SITE' not in self.env:
 
  32             self.env['PYTHON_SITE'] = os.path.join(
 
  33                         self.env['PYTHON_BASE'], 'lib',
 
  34                         'python' + self.env['PYTHON_VERSION'], 'site-packages')
 
  36         if 'APP_DIR' not in self.env:
 
  37             self.env['APP_DIR'] = os.path.join(self.env['ROOT'], 'application')
 
  39         if 'CONFIG_DIR' not in self.env:
 
  40             self.env['CONFIG_DIR'] = os.path.join(self.env['ROOT'], 'etc')
 
  42         if 'MEDIA_DIR' not in self.env:
 
  43             self.env['MEDIA_DIR'] = os.path.join(self.env['ROOT'], 'www', 'media')
 
  45         self._logger = logging.getLogger("deployment")
 
  47     def info(self, *args, **kwargs):
 
  48         self._logger.info(*args, **kwargs)
 
  50     def render_template(self, source, dest, extra_context={}):
 
  51         self.info("Rendering template: %s", source)
 
  53         with open(source, 'rb') as source_file:
 
  54             t = Template(source_file.read())
 
  56         context = dict(self.env)
 
  57         context.update(extra_context)
 
  59         with open(dest, 'wb') as dest_file:
 
  60             dest_file.write(t.safe_substitute(context))
 
  64     def restart_app(self):
 
  70     def update_config(self):
 
  73     def install_dependencies(self):
 
  78         self.install_dependencies()
 
  82     def find_resource(self, path):
 
  83         for dir in (self.env['CONFIG_DIR'], self.env['APP_DIR']):
 
  84             full_path = os.path.join(dir, path)
 
  85             if os.path.isfile(full_path):
 
  88         raise ValueError("Resource '%s' not found" % path)
 
  91     def run_deploy(cls, *args, **kwargs):
 
  92         site = cls(*args, **kwargs)
 
  95 class WSGISite(DeploySite):
 
  97     def __init__(self, **env):
 
  98         super(WSGISite, self).__init__(**env)
 
 100         if 'WSGI_FILE' not in self.env:
 
 101             self.env['WSGI_FILE'] = os.path.join(self.env['ROOT'], 'www',
 
 102                                         'wsgi', self.env['PROJECT_NAME']) + '.wsgi'
 
 104         self.env['WSGI_DIR'] = os.path.dirname(self.env['WSGI_FILE'])
 
 106         if 'WSGI_SOURCE_FILE' not in self.env:
 
 107             self.env['WSGI_SOURCE_FILE'] = 'wsgi_app.template'
 
 109         if 'WSGI_USER' not in self.env:
 
 110             self.env['WSGI_USER'] = 'www-data'
 
 112     def restart_app(self):
 
 113         self.info("Restarting wsgi application: %s", self.env['WSGI_FILE'])
 
 114         os.system("touch %s" % self.env['WSGI_FILE'])
 
 116     def update_config(self):
 
 117         super(WSGISite, self).update_config()
 
 119         source = self.find_resource(self.env['WSGI_SOURCE_FILE'])
 
 120         self.render_template(source, self.env['WSGI_FILE'])
 
 122 class PIPSite(DeploySite):
 
 124     def install_dependencies(self):
 
 125         self.info("Installing requirements")
 
 126         os.system("%s install -r %s" % (self.env['PIP_BIN'], self.find_resource('requirements.txt')))
 
 129             self.info("Installing local requirements")
 
 130             os.system("%s install -r %s" % (self.env['PIP_BIN'], self.find_resource('requirements_local.txt')))
 
 134 class GitSite(DeploySite):
 
 136     def update_app(self):
 
 137         self.info("Updating repository.")
 
 138         os.system("cd %s; git pull" % self.env['APP_DIR'])
 
 140 class ApacheSite(DeploySite):
 
 142     def __init__(self, **env):
 
 143         super(ApacheSite, self).__init__(**env)
 
 145         if 'VHOST_SOURCE_FILE' not in self.env:
 
 146             self.env['VHOST_SOURCE_FILE'] = 'apache_vhost.template'
 
 148         if 'VHOST_FILE' not in self.env:
 
 149             self.env['VHOST_FILE'] = os.path.join(self.env['CONFIG_DIR'], self.env['PROJECT_NAME'] + '.vhost')
 
 151     def update_config(self):
 
 152         super(ApacheSite, self).update_config()
 
 154         source = self.find_resource(self.env['VHOST_SOURCE_FILE'])
 
 155         self.render_template(source, self.env['VHOST_FILE'])