# Copyright (C) 2012-2015 Codethink Limited # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; version 2 of the License. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License along # with this program. If not, see . import logging import os import shutil import stat import cliapp from urlparse import urlparse import tempfile import morphlib class StagingArea(object): '''Represent the staging area for building software. The staging area is a temporary directory. In normal operation the build dependencies of the artifact being built are installed into the staging area and then 'chroot' is used to isolate the build processes from the host system. Chunks built in 'test' or 'build-essential' mode have an empty staging area and are allowed to use the tools of the host. ''' _base_path = ['/sbin', '/usr/sbin', '/bin', '/usr/bin'] def __init__(self, app, dirname, build_env, use_chroot=True, extra_env={}, extra_path=[]): self._app = app self.dirname = dirname self.builddirname = None self.destdirname = None self._bind_readonly_mount = None self.use_chroot = use_chroot self.env = build_env.env self.env.update(extra_env) if use_chroot: path = extra_path + build_env.extra_path + self._base_path else: rel_path = extra_path + build_env.extra_path full_path = [os.path.normpath(dirname + p) for p in rel_path] path = full_path + os.environ['PATH'].split(':') self.env['PATH'] = ':'.join(path) # Wrapper to be overridden by unit tests. def _mkdir(self, dirname): # pragma: no cover os.makedirs(dirname) def _dir_for_source(self, source, suffix): dirname = os.path.join(self.dirname, '%s.%s' % (str(source.name), suffix)) self._mkdir(dirname) return dirname def builddir(self, source): '''Create a build directory for a given source project. Return path to directory. ''' return self._dir_for_source(source, 'build') def destdir(self, source): '''Create an installation target directory for a given source project. This is meant to be used as $DESTDIR when installing chunks. Return path to directory. ''' return self._dir_for_source(source, 'inst') def overlay_upperdir(self, source): '''Create a directory to be upperdir for overlayfs, and return it.''' return self._dir_for_source(source, 'overlay_upper') def overlaydir(self, source): '''Create a directory to be a mount point for overlayfs, return it''' return self._dir_for_source(source, 'overlay') def relative(self, filename): '''Return a filename relative to the staging area.''' if not self.use_chroot: return filename dirname = self.dirname if not dirname.endswith('/'): dirname += '/' assert filename.startswith(dirname) return filename[len(dirname) - 1:] # include leading slash def create_devices(self, morphology): # pragma: no cover '''Creates device nodes if the morphology specifies them''' perms_mask = stat.S_IRWXU | stat.S_IRWXG | stat.S_IRWXO if 'devices' in morphology and morphology['devices'] is not None: for dev in morphology['devices']: destfile = os.path.join(self.dirname, './' + dev['filename']) mode = int(dev['permissions'], 8) & perms_mask if dev['type'] == 'c': mode = mode | stat.S_IFCHR elif dev['type'] == 'b': mode = mode | stat.S_IFBLK else: raise IOError('Cannot create device node %s,' 'unrecognized device type "%s"' % (destfile, dev['type'])) parent = os.path.dirname(destfile) if not os.path.exists(parent): os.makedirs(parent) if not os.path.exists(destfile): logging.debug("Creating device node %s" % destfile) os.mknod(destfile, mode, os.makedev(dev['major'], dev['minor'])) os.chown(destfile, dev['uid'], dev['gid']) def install_artifact(self, artifact_cache, artifact): '''Install a build artifact into the staging area.''' if not os.path.exists(self.dirname): self._mkdir(self.dirname) artifact_cache.get(artifact, directory=self.dirname) self.create_devices(artifact.source.morphology) def remove(self): '''Remove the entire staging area. Do not expect anything with the staging area to work after this method is called. Be careful about calling this method if the filesystem root directory was given as the dirname. ''' shutil.rmtree(self.dirname) to_mount_in_staging = ( ('dev/shm', 'tmpfs', 'none'), ) to_mount_in_bootstrap = () def ccache_dir(self, source): #pragma: no cover ccache_dir = self._app.settings['compiler-cache-dir'] if not os.path.isdir(ccache_dir): os.makedirs(ccache_dir) # Get a path for the repo's ccache ccache_url = source.repo.url ccache_path = urlparse(ccache_url).path ccache_repobase = os.path.basename(ccache_path) if ':' in ccache_repobase: # the basename is a repo-alias resolver = morphlib.repoaliasresolver.RepoAliasResolver( self._app.settings['repo-alias']) ccache_url = resolver.pull_url(ccache_repobase) ccache_path = urlparse(ccache_url).path ccache_repobase = os.path.basename(ccache_path) if ccache_repobase.endswith('.git'): ccache_repobase = ccache_repobase[:-len('.git')] ccache_repodir = os.path.join(ccache_dir, ccache_repobase) # Make sure that directory exists if not os.path.isdir(ccache_repodir): os.mkdir(ccache_repodir) # Get the destination path ccache_destdir= os.path.join(self.dirname, 'tmp', 'ccache') # Make sure that the destination exists. We'll create /tmp if necessary # to avoid breaking when faced with an empty staging area. if not os.path.isdir(ccache_destdir): os.makedirs(ccache_destdir) return ccache_repodir def chroot_open(self, source, setup_mounts): # pragma: no cover '''Setup staging area for use as a chroot.''' assert self.builddirname == None and self.destdirname == None builddir = self.builddir(source) destdir = self.destdir(source) self.builddirname = builddir self.destdirname = destdir return builddir, destdir def chroot_close(self): # pragma: no cover '''Undo changes by chroot_open. This should be called after the staging area is no longer needed. ''' # No cleanup is currently required pass def runcmd(self, argv, **kwargs): # pragma: no cover '''Run a command in a chroot in the staging area.''' assert 'env' not in kwargs kwargs['env'] = dict(self.env) if 'extra_env' in kwargs: kwargs['env'].update(kwargs['extra_env']) del kwargs['extra_env'] ccache_dir = kwargs.pop('ccache_dir', None) chroot_dir = self.dirname if self.use_chroot else '/' temp_dir = kwargs["env"].get("TMPDIR", "/tmp") staging_dirs = [self.builddirname, self.destdirname] if self.use_chroot: staging_dirs += ["dev", "proc", temp_dir.lstrip('/')] do_not_mount_dirs = [os.path.join(self.dirname, d) for d in staging_dirs] if not self.use_chroot: do_not_mount_dirs += [temp_dir] logging.debug("Not mounting dirs %r" % do_not_mount_dirs) if self.use_chroot: mounts = self.to_mount_in_staging else: mounts = [(os.path.join(self.dirname, target), type, source) for target, type, source in self.to_mount_in_bootstrap] mount_proc = self.use_chroot if ccache_dir and not self._app.settings['no-ccache']: ccache_target = os.path.join( self.dirname, kwargs['env']['CCACHE_DIR'].lstrip('/')) binds = ((ccache_dir, ccache_target),) else: binds = () container_config=dict( cwd=kwargs.pop('cwd', '/'), root=chroot_dir, mounts=mounts, mount_proc=mount_proc, binds=binds, writable_paths=do_not_mount_dirs) cmdline = morphlib.util.containerised_cmdline( argv, **container_config) if kwargs.get('logfile') != None: logfile = kwargs.pop('logfile') teecmd = ['tee', '-a', logfile] exit, out, err = self._app.runcmd_unchecked( cmdline, teecmd, **kwargs) else: exit, out, err = self._app.runcmd_unchecked(cmdline, **kwargs) if exit == 0: return out else: logging.debug('Command returned code %i', exit) msg = morphlib.util.error_message_for_containerised_commandline( argv, err, container_config) raise cliapp.AppException( 'In staging area %s: %s' % (self._failed_location(), msg)) def _failed_location(self): # pragma: no cover '''Path this staging area will be moved to if an error occurs.''' return os.path.join(self._app.settings['tempdir'], 'failed', os.path.basename(self.dirname)) def abort(self): # pragma: no cover '''Handle what to do with a staging area in the case of failure. This may either remove it or save it for later inspection. ''' # TODO: when we add the option to throw away failed builds, # hook it up here dest_dir = self._failed_location() os.rename(self.dirname, dest_dir) self.dirname = dest_dir