110 lines
4.1 KiB
Python
110 lines
4.1 KiB
Python
|
"""
|
||
|
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 <http://www.gnu.org/licenses/>.
|
||
|
"""
|
||
|
import logging
|
||
|
import json
|
||
|
import os
|
||
|
import tarfile
|
||
|
import tempfile
|
||
|
import filecmp
|
||
|
import shutil
|
||
|
import pmb.build
|
||
|
import pmb.parse.apkbuild
|
||
|
|
||
|
|
||
|
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 = tar_b.getnames()
|
||
|
list_b.sort()
|
||
|
if list_a != list_b:
|
||
|
raise RuntimeError(
|
||
|
"Both APKs do not contain the same file names!")
|
||
|
|
||
|
# Iterate through the list
|
||
|
for name in list_a:
|
||
|
logging.debug("Compare: " + name)
|
||
|
if name == ".PKGINFO" or name.startswith(".SIGN.RSA."):
|
||
|
logging.debug(
|
||
|
"=> Skipping, this is expected to be different")
|
||
|
continue
|
||
|
temp_files = []
|
||
|
|
||
|
# Extract
|
||
|
for tar in [tar_a, tar_b]:
|
||
|
member = tar.getmember(name)
|
||
|
if member.isdir():
|
||
|
continue
|
||
|
handle, path = tempfile.mkstemp("pmbootstrap")
|
||
|
handle = open(handle, "wb")
|
||
|
shutil.copyfileobj(tar.extractfile(member), handle)
|
||
|
handle.close()
|
||
|
temp_files.append(path)
|
||
|
if not len(temp_files):
|
||
|
logging.debug("=> Skipping, this is a directory")
|
||
|
continue
|
||
|
|
||
|
# 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("=> Equal")
|
||
|
else:
|
||
|
raise RuntimeError("File '" + name + "' is different!")
|
||
|
|
||
|
|
||
|
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 <pkgname>'")
|
||
|
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)
|