""" Copyright 2017 Oliver Smith This file is part of pmbootstrap. pmbootstrap 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, either version 3 of the License, or (at your option) any later version. pmbootstrap 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 pmbootstrap. If not, see . """ import logging import json import os import tarfile import tempfile import filecmp import shutil import pmb.build import pmb.parse.apkbuild def diff_files(tar_a, tar_b, member_a, member_b, name): # Extract both files tars = [tar_a, tar_b] members = [member_a, member_b] temp_files = [] for i in range(2): handle, path = tempfile.mkstemp("pmbootstrap") handle = open(handle, "wb") shutil.copyfileobj(tars[i].extractfile(members[i]), handle) handle.close() temp_files.append(path) # Compare and delete equal = filecmp.cmp(temp_files[0], temp_files[1], shallow=False) for temp_file in temp_files: os.remove(temp_file) if equal: logging.debug("=> File has the same content") else: raise RuntimeError("File '" + name + "' is different!") def diff(args, apk_a, apk_b): logging.info("Challenge " + apk_a) with tarfile.open(apk_a, "r:gz") as tar_a: with tarfile.open(apk_b, "r:gz") as tar_b: # List of files must be the same list_a = sorted(tar_a.getnames()) list_b = sorted(tar_b.getnames()) if list_a != list_b: raise RuntimeError( "Both APKs do not contain the same file names!") # Iterate through the list success = True for name in list_a: try: logging.debug("Compare: " + name) if name == ".PKGINFO" or name.startswith(".SIGN.RSA."): logging.debug( "=> Skipping, this is expected to be different") continue # Get members member_a = tar_a.getmember(name) member_b = tar_b.getmember(name) if member_a.type != member_b.type: raise RuntimeError( "Entry '" + name + "' has a different type!") if member_a.isdir(): logging.debug("=> Skipping, this is directory") elif member_a.isfile(): diff_files(tar_a, tar_b, member_a, member_b, name) elif member_a.issym() or member_a.islnk(): if member_a.linkname == member_b.linkname: logging.debug( "=> Both link to " + member_a.linkname) else: raise RuntimeError( "Link " + name + " has a different target!") else: raise RuntimeError( "Can't diff '" + name + "', unsupported type!") except Exception as e: logging.info("CHALLENGE FAILED for " + name + ":" + str(e)) success = False if not success: raise RuntimeError("Challenge failed (see errors above)") def challenge(args, apk_path): # Parse buildinfo buildinfo_path = apk_path + ".buildinfo.json" if not os.path.exists(buildinfo_path): logging.info("NOTE: To create a .buildinfo.json file, use the" " --buildinfo command while building: 'pmbootstrap build" " --buildinfo '") raise RuntimeError("Missing file: " + buildinfo_path) with open(buildinfo_path) as handle: buildinfo = json.load(handle) # Parse and install all packages listed in versions versions = {} for package in buildinfo["versions"]: split = pmb.chroot.apk.package_split(package) pkgname = split["pkgname"] versions[pkgname] = split pmb.chroot.apk.install(args, versions.keys()) # Verify the installed versions installed = pmb.chroot.apk.installed(args) for pkgname, split in versions.items(): package_installed = installed[pkgname]["package"] package_buildinfo = split["package"] if package_installed != package_buildinfo: raise RuntimeError("Dependency " + pkgname + " version is different" " (installed: " + package_installed + "," " buildinfo: " + package_buildinfo + ")!") # Build the package output = pmb.build.package(args, buildinfo["pkgname"], buildinfo["carch"], force=True) # Diff the apk contents diff(args, apk_path, args.work + "/packages/" + output)