2017-01-18 17:43:46 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
2017-01-24 23:15:13 +01:00
|
|
|
import argparse, sys, shutil, os, hashlib
|
2017-01-19 17:39:55 +01:00
|
|
|
|
2017-01-19 22:56:17 +01:00
|
|
|
sys.path.append(os.path.dirname(__file__) + '/lib')
|
2017-01-19 17:39:55 +01:00
|
|
|
|
2017-01-24 17:32:19 +01:00
|
|
|
import tamarin, system, rkt
|
2017-01-19 17:39:55 +01:00
|
|
|
|
2017-02-11 11:51:01 +01:00
|
|
|
def create_args_parser():
|
|
|
|
'''Return a new configured ArgumentParser'''
|
2017-01-19 22:56:17 +01:00
|
|
|
profile_names = tamarin.get_available_profile_names()
|
2017-01-19 17:39:55 +01:00
|
|
|
|
2017-02-09 21:53:24 +01:00
|
|
|
parser = argparse.ArgumentParser(description="Generate packages for various GNU/Linux distributions")
|
2017-01-19 17:39:55 +01:00
|
|
|
|
|
|
|
# Define available/required arguments and flags
|
2017-02-09 21:53:24 +01:00
|
|
|
parser.add_argument("project_directory", help="The path to your project to package")
|
|
|
|
parser.add_argument("-o", "--output", help="The path to the generated packages destination directory", default=".")
|
2017-01-19 22:56:17 +01:00
|
|
|
parser.add_argument("-p", "--profile", help="The profile to use to package this project (default: debian)", choices=profile_names, default='debian')
|
2017-02-09 21:53:24 +01:00
|
|
|
parser.add_argument("-a", "--architecture", help="The target architecture for the package (default: amd64)", default='amd64')
|
2017-01-24 23:15:13 +01:00
|
|
|
parser.add_argument("--rebuild", help="Ignore cache and rebuild container's image", action="store_true", default=False)
|
2017-01-19 17:39:55 +01:00
|
|
|
|
|
|
|
return parser
|
|
|
|
|
2017-02-11 11:51:01 +01:00
|
|
|
def download_and_extract_rkt(dest_dir, verbose=True):
|
|
|
|
'''Download and extract rkt to the given destination directory'''
|
|
|
|
rkt_archive_path = tamarin.download_rkt()
|
|
|
|
system.extract_tar(rkt_archive_path, workspace_tmp)
|
|
|
|
rkt_archive_dir = tamarin.get_rkt_achive_dest_dir()
|
|
|
|
shutil.rmtree(local_rkt_dir, ignore_errors=True)
|
|
|
|
os.rename(rkt_archive_dir, dest_dir)
|
|
|
|
|
|
|
|
def download_and_extract_acbuild(dest_dir, verbose=True):
|
|
|
|
'''Download and extract acbuild to the given destination directory'''
|
|
|
|
acbuild_archive_path = tamarin.download_acbuild()
|
|
|
|
system.extract_tar(acbuild_archive_path, workspace_tmp)
|
|
|
|
acbuild_archive_dir = tamarin.get_acbuild_achive_dest_dir()
|
|
|
|
shutil.rmtree(local_acbuild_dir, ignore_errors=True)
|
|
|
|
os.rename(acbuild_archive_dir, dest_dir)
|
|
|
|
|
|
|
|
def get_cached_image_path(profile):
|
|
|
|
'''Compute and return the path for an hypothetic cached image for the given profile'''
|
|
|
|
containerbuild_hooks = profile['containerbuild']['hooks']
|
|
|
|
hasher = hashlib.sha1()
|
|
|
|
hasher.update(base_image.encode())
|
|
|
|
hasher.update(containerbuild_hooks.encode())
|
|
|
|
image_hash = hasher.hexdigest()
|
|
|
|
cache_dir = tamarin.get_workspace_subdir('cache')
|
|
|
|
return os.path.join(os.sep, cache_dir, '{:s}.aci'.format(image_hash[:12]));
|
|
|
|
|
|
|
|
def build_image(build_workspace, aci_file, base_image, profile):
|
|
|
|
|
|
|
|
acbuild_flags = ["--modify", aci_file, "--work-path", build_workspace]
|
|
|
|
|
|
|
|
# Find and export base image from rkt' store
|
|
|
|
name_pattern = base_image.split('/')[-1] + '$'
|
|
|
|
image = rkt.find_image_by_name(name_pattern, rkt_flags=rkt_flags)
|
|
|
|
rkt.export_image(image['id'], aci_file, rkt_flags=rkt_flags);
|
|
|
|
|
|
|
|
# Build image
|
|
|
|
tamarin.run_acbuild(acbuild_flags+["set-name", "image_{:d}".format(pid)])
|
|
|
|
tamarin.run_acbuild(acbuild_flags+["mount", "add", "src", "/src", "--read-only"])
|
|
|
|
tamarin.run_acbuild(acbuild_flags+["mount", "add", "dist", "/dist"])
|
|
|
|
tamarin.run_acbuild(acbuild_flags+["mount", "add", "tamarin-hooks", "/tamarin/hooks", "--read-only"])
|
|
|
|
tamarin.run_acbuild(acbuild_flags+["mount", "add", "tamarin-lib", "/tamarin/lib", "--read-only"])
|
|
|
|
tamarin.run_acbuild(acbuild_flags+["mount", "add", "tamarin-profiles", "/tamarin/profiles", "--read-only"])
|
|
|
|
|
|
|
|
# Configure "containerbuild" hooks environment
|
|
|
|
hooks_env = os.environ.copy()
|
|
|
|
hooks_env["PATH"] = os.environ['PATH'] + ':' + tamarin.get_workspace_subdir('acbuild')
|
|
|
|
hooks_env["TAMARIN_ACBUILD"] = " ".join([system.which('acbuild', local_acbuild_dir)]+acbuild_flags)
|
|
|
|
hooks_env["TAMARIN_ACBUILD_ENGINE"] = "chroot" if not system.which('systemctl') else "systemd-nspawn"
|
|
|
|
|
|
|
|
# Run hooks
|
|
|
|
tamarin.run_profile_hooks(profile, 'containerbuild', cwd=build_workspace, env=hooks_env)
|
|
|
|
|
|
|
|
return aci_file
|
|
|
|
|
|
|
|
def cleanup(build_workspace, rkt_flags):
|
|
|
|
|
|
|
|
# Nettoyage des conteneurs
|
|
|
|
rkt.run([
|
|
|
|
"gc",
|
|
|
|
"--grace-period=0"
|
|
|
|
] + rkt_flags, as_root=True)
|
|
|
|
|
|
|
|
# Nettoyage des images obsolètes du store
|
|
|
|
rkt.run([
|
|
|
|
"image",
|
|
|
|
"gc"
|
|
|
|
] + rkt_flags, as_root=True)
|
|
|
|
|
|
|
|
# Suppression de l'espace de travail de build
|
|
|
|
shutil.rmtree(build_workspace, ignore_errors=True)
|
|
|
|
|
|
|
|
def validate_args(args):
|
|
|
|
'''TODO'''
|
|
|
|
|
2017-01-18 17:43:46 +01:00
|
|
|
if __name__ == "__main__":
|
|
|
|
|
2017-02-11 11:51:01 +01:00
|
|
|
parser = create_args_parser()
|
2017-01-19 22:56:17 +01:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
2017-02-11 11:51:01 +01:00
|
|
|
validate_args(args)
|
|
|
|
|
2017-02-09 21:53:24 +01:00
|
|
|
# Verify project directory
|
|
|
|
project_dir = os.path.abspath(args.project_directory)
|
|
|
|
output_dir = os.path.abspath(args.output)
|
|
|
|
|
2017-01-24 23:15:13 +01:00
|
|
|
# Load build profile
|
2017-01-19 22:56:17 +01:00
|
|
|
profile = tamarin.load_profile(args.profile)
|
|
|
|
|
|
|
|
workspace = tamarin.get_workspace_dir()
|
|
|
|
workspace_tmp = tamarin.get_workspace_subdir('tmp')
|
|
|
|
|
|
|
|
local_rkt_dir = tamarin.get_workspace_subdir('rkt')
|
|
|
|
if not system.which('rkt', local_rkt_dir):
|
2017-02-11 11:51:01 +01:00
|
|
|
download_and_extract_rkt(local_rkt_dir)
|
2017-01-19 22:56:17 +01:00
|
|
|
|
|
|
|
local_acbuild_dir = tamarin.get_workspace_subdir('acbuild')
|
|
|
|
if not system.which('acbuild', local_acbuild_dir):
|
2017-02-11 11:51:01 +01:00
|
|
|
download_and_extract_acbuild(local_acbuild_dir)
|
2017-01-19 22:56:17 +01:00
|
|
|
|
|
|
|
pid = os.getpid()
|
2017-01-24 23:15:13 +01:00
|
|
|
build_workspace = tamarin.get_workspace_subdir('tmp/build_{:d}'.format(pid))
|
2017-01-19 22:56:17 +01:00
|
|
|
|
2017-01-24 23:15:13 +01:00
|
|
|
rkt_store = tamarin.get_workspace_subdir('store')
|
|
|
|
rkt_flags = ["--dir={:s}".format(rkt_store)]
|
2017-01-20 17:22:54 +01:00
|
|
|
|
|
|
|
base_image = profile['profile']['default_image']
|
2017-02-11 11:51:01 +01:00
|
|
|
|
|
|
|
# If the base image is Docker-based, download it
|
2017-01-20 17:22:54 +01:00
|
|
|
if base_image.startswith('docker://'):
|
2017-01-24 17:32:19 +01:00
|
|
|
rkt.run([
|
2017-01-20 17:22:54 +01:00
|
|
|
"fetch",
|
|
|
|
"--insecure-options=image",
|
|
|
|
base_image
|
|
|
|
] + rkt_flags)
|
2017-01-24 17:32:19 +01:00
|
|
|
|
2017-01-24 23:15:13 +01:00
|
|
|
aci_file = os.path.join(os.sep, build_workspace, 'image.aci')
|
2017-02-11 11:51:01 +01:00
|
|
|
cached_image_file = get_cached_image_path(profile)
|
2017-01-24 23:15:13 +01:00
|
|
|
|
|
|
|
if not args.rebuild and os.path.exists(cached_image_file):
|
2017-02-11 11:51:01 +01:00
|
|
|
# Copy cached image
|
|
|
|
shutil.copyfile(cached_image_file, aci_file)
|
2017-01-24 23:15:13 +01:00
|
|
|
else:
|
2017-02-11 11:51:01 +01:00
|
|
|
build_image(build_workspace, aci_file, base_image, profile)
|
|
|
|
# Cache image
|
|
|
|
shutil.copyfile(aci_file, cached_image_file)
|
2017-01-24 23:15:13 +01:00
|
|
|
|
|
|
|
# Start container
|
2017-01-25 14:49:30 +01:00
|
|
|
rkt.run(rkt_flags+[
|
2017-01-20 17:22:54 +01:00
|
|
|
"run",
|
|
|
|
"--insecure-options=image",
|
2017-01-25 14:49:30 +01:00
|
|
|
aci_file, "--net=host",
|
2017-02-09 21:53:24 +01:00
|
|
|
"--volume=src,kind=host,source={:s}".format(project_dir),
|
|
|
|
"--volume=dist,kind=host,source={:s}".format(output_dir),
|
|
|
|
"--volume=tamarin-hooks,kind=host,source={:s}".format(tamarin.get_hooks_dir()),
|
|
|
|
"--volume=tamarin-lib,kind=host,source={:s}".format(tamarin.get_lib_dir()),
|
|
|
|
"--volume=tamarin-profiles,kind=host,source={:s}".format(tamarin.get_profiles_dir()),
|
2017-02-10 15:31:42 +01:00
|
|
|
"--set-env=HTTP_PROXY={:s}".format(os.environ['HTTP_PROXY']),
|
|
|
|
"--set-env=HTTPS_PROXY={:s}".format(os.environ['HTTP_PROXY']),
|
|
|
|
"--set-env=http_proxy={:s}".format(os.environ['http_proxy']),
|
|
|
|
"--set-env=https_proxy={:s}".format(os.environ['https_proxy']),
|
2017-02-09 22:38:04 +01:00
|
|
|
#"--interactive", "--exec", "/bin/bash"
|
|
|
|
"--exec", "/usr/bin/python3", "--", "/tamarin/lib/build.py", args.profile, args.architecture
|
2017-01-25 14:49:30 +01:00
|
|
|
], as_root=True)
|
2017-01-20 17:22:54 +01:00
|
|
|
|
|
|
|
# Cleanup
|
2017-02-11 11:51:01 +01:00
|
|
|
cleanup(build_workspace, rkt_flags)
|