# 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 fcntl import pipes 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, source, dirname, build_env, use_chroot=True, extra_env={}, extra_path=[]): self._app = app self.source = source self.dirname = dirname self.use_chroot = use_chroot self.env = build_env.env self.env.update(extra_env) os.makedirs(self.real_builddir()) os.makedirs(self.real_destdir()) 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) # Keep trying until we have created a directory with an # exclusive lock on it, as if the user runs `morph gc` in # parallel the staging area directory could have been removed # or have its exclusive lock associated with the `morph gc` # process while True: try: fd = os.open(dirname, os.O_RDONLY) fcntl.flock(fd, fcntl.LOCK_EX) if os.path.exists(dirname): self.staging_area_fd = fd break else: os.close(fd) # pragma: no cover except OSError: # pragma: no cover if not os.path.exists(dirname): os.makedirs(dirname) def relative(self, path): '''Return a path relative to the staging area.''' if self.use_chroot: return os.path.join(os.sep, path) else: return os.path.join(self.dirname, path) def relative_builddir(self): return self.relative('%s.build' % self.source.name) def relative_destdir(self): return self.relative('%s.inst' % self.source.name) def real_builddir(self): '''Build directory for a given source project ''' return os.path.join(self.dirname, '%s.build' % (self.source.name)) def real_destdir(self): '''Installation target directory for a given source project. This is meant to be used as $DESTDIR when installing chunks. ''' return os.path.join(self.dirname, '%s.inst' % (self.source.name)) def hardlink_all_files(self, srcpath, destpath): # pragma: no cover '''Hardlink every file in the path to the staging-area If an exception is raised, the staging-area is indeterminate. ''' file_stat = os.lstat(srcpath) mode = file_stat.st_mode if stat.S_ISDIR(mode): # Ensure directory exists in destination, then recurse. if not os.path.lexists(destpath): os.makedirs(destpath) dest_stat = os.stat(os.path.realpath(destpath)) if not stat.S_ISDIR(dest_stat.st_mode): raise IOError('Destination not a directory. source has %s' ' destination has %s' % (srcpath, destpath)) for entry in os.listdir(srcpath): self.hardlink_all_files(os.path.join(srcpath, entry), os.path.join(destpath, entry)) elif stat.S_ISLNK(mode): # Copy the symlink. if os.path.lexists(destpath): os.remove(destpath) os.symlink(os.readlink(srcpath), destpath) elif stat.S_ISREG(mode): # Hardlink the file. if os.path.lexists(destpath): os.remove(destpath) os.link(srcpath, destpath) elif stat.S_ISCHR(mode) or stat.S_ISBLK(mode): # Block or character device. Put contents of st_dev in a mknod. if os.path.lexists(destpath): os.remove(destpath) os.mknod(destpath, file_stat.st_mode, file_stat.st_rdev) os.chmod(destpath, file_stat.st_mode) else: # Unsupported type. raise IOError('Cannot extract %s into staging-area. Unsupported' ' type.' % srcpath) def install_artifact(self, handle): '''Install a build artifact into the staging area. We access the artifact via an open file handle. For now, we assume the artifact is a tarball. ''' chunk_cache_dir = os.path.join(self._app.settings['tempdir'], 'chunks') unpacked_artifact = os.path.join( chunk_cache_dir, os.path.basename(handle.name) + '.d') if not os.path.exists(unpacked_artifact): self._app.status( msg='Unpacking chunk from cache %(filename)s', filename=os.path.basename(handle.name)) with morphlib.util.temp_dir(dir=chunk_cache_dir, cleanup_on_success=False) as savedir: morphlib.bins.unpack_binary_from_file( handle, savedir + '/') # TODO: This rename is not concurrency safe if two builds are # extracting the same chunk, one build will fail because # the other renamed its tempdir here first. os.rename(savedir, unpacked_artifact) self.hardlink_all_files(unpacked_artifact, self.dirname) 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) os.close(self.staging_area_fd) to_mount_in_staging = ( ('dev/shm', 'tmpfs', 'none'), ) to_mount_in_bootstrap = () def ccache_dir(self): #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 = self.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 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.real_builddir(), self.real_destdir()] 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: logging.debug('Command returned code %i', exit) chroot_script = self.dirname + '.sh' shell_command = ['env', '-i', '--'] for k, v in kwargs['env'].iteritems(): shell_command += ["%s=%s" % (k, v)] shell_command += [os.path.join(os.sep, 'bin', 'sh')] cmdline = morphlib.util.containerised_cmdline( shell_command, **container_config) with open(chroot_script, 'w') as f: f.write(' '.join(map(pipes.quote, cmdline))) return exit