|
|
@ -26,7 +26,6 @@ import shutil
|
|
|
|
import subprocess
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import sys
|
|
|
|
import tempfile
|
|
|
|
import tempfile
|
|
|
|
from collections import namedtuple
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
from libcalamares import *
|
|
|
|
from libcalamares import *
|
|
|
|
|
|
|
|
|
|
|
@ -59,10 +58,13 @@ def list_excludes(destination):
|
|
|
|
"""
|
|
|
|
"""
|
|
|
|
lst = []
|
|
|
|
lst = []
|
|
|
|
extra_mounts = globalstorage.value("extraMounts")
|
|
|
|
extra_mounts = globalstorage.value("extraMounts")
|
|
|
|
|
|
|
|
|
|
|
|
for extra_mount in extra_mounts:
|
|
|
|
for extra_mount in extra_mounts:
|
|
|
|
mount_point = extra_mount["mountPoint"]
|
|
|
|
mount_point = extra_mount["mountPoint"]
|
|
|
|
|
|
|
|
|
|
|
|
if mount_point:
|
|
|
|
if mount_point:
|
|
|
|
lst.extend(['--exclude', mount_point + '/'])
|
|
|
|
lst.extend(['--exclude', mount_point + '/'])
|
|
|
|
|
|
|
|
|
|
|
|
return lst
|
|
|
|
return lst
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -87,11 +89,7 @@ def file_copy(source, dest, progress_cb):
|
|
|
|
args = ['rsync', '-aHAXr']
|
|
|
|
args = ['rsync', '-aHAXr']
|
|
|
|
args.extend(list_excludes(dest))
|
|
|
|
args.extend(list_excludes(dest))
|
|
|
|
args.extend(['--progress', source, dest])
|
|
|
|
args.extend(['--progress', source, dest])
|
|
|
|
process = subprocess.Popen(args,
|
|
|
|
process = subprocess.Popen(args, env=at_env, bufsize=1, stdout=subprocess.PIPE, close_fds=ON_POSIX)
|
|
|
|
env=at_env,
|
|
|
|
|
|
|
|
bufsize=1,
|
|
|
|
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
|
|
|
|
close_fds=ON_POSIX)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for line in iter(process.stdout.readline, b''):
|
|
|
|
for line in iter(process.stdout.readline, b''):
|
|
|
|
# small comment on this regexp.
|
|
|
|
# small comment on this regexp.
|
|
|
@ -107,6 +105,7 @@ def file_copy(source, dest, progress_cb):
|
|
|
|
# therefore we can easily subtract x from y in order to get real files
|
|
|
|
# therefore we can easily subtract x from y in order to get real files
|
|
|
|
# copied / processed count.
|
|
|
|
# copied / processed count.
|
|
|
|
m = re.findall(r'xfr#(\d+), ir-chk=(\d+)/(\d+)', line.decode())
|
|
|
|
m = re.findall(r'xfr#(\d+), ir-chk=(\d+)/(\d+)', line.decode())
|
|
|
|
|
|
|
|
|
|
|
|
if m:
|
|
|
|
if m:
|
|
|
|
# we've got a percentage update
|
|
|
|
# we've got a percentage update
|
|
|
|
num_files_remaining = int(m[0][1])
|
|
|
|
num_files_remaining = int(m[0][1])
|
|
|
@ -117,9 +116,12 @@ def file_copy(source, dest, progress_cb):
|
|
|
|
# I guess we're updating every 100 files...
|
|
|
|
# I guess we're updating every 100 files...
|
|
|
|
if num_files_copied % 100 == 0:
|
|
|
|
if num_files_copied % 100 == 0:
|
|
|
|
progress_cb(num_files_copied)
|
|
|
|
progress_cb(num_files_copied)
|
|
|
|
|
|
|
|
|
|
|
|
process.wait()
|
|
|
|
process.wait()
|
|
|
|
|
|
|
|
|
|
|
|
if process.returncode != 0:
|
|
|
|
if process.returncode != 0:
|
|
|
|
return "rsync failed with error code {}.".format(process.returncode)
|
|
|
|
return "rsync failed with error code {}.".format(process.returncode)
|
|
|
|
|
|
|
|
|
|
|
|
return None
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -128,6 +130,7 @@ class UnpackOperation:
|
|
|
|
|
|
|
|
|
|
|
|
:param entries:
|
|
|
|
:param entries:
|
|
|
|
"""
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, entries):
|
|
|
|
def __init__(self, entries):
|
|
|
|
self.entries = entries
|
|
|
|
self.entries = entries
|
|
|
|
self.entry_for_source = dict((x.source, x) for x in self.entries)
|
|
|
|
self.entry_for_source = dict((x.source, x) for x in self.entries)
|
|
|
@ -135,6 +138,7 @@ class UnpackOperation:
|
|
|
|
def report_progress(self):
|
|
|
|
def report_progress(self):
|
|
|
|
""" Pass progress to user interface """
|
|
|
|
""" Pass progress to user interface """
|
|
|
|
progress = float(0)
|
|
|
|
progress = float(0)
|
|
|
|
|
|
|
|
|
|
|
|
for entry in self.entries:
|
|
|
|
for entry in self.entries:
|
|
|
|
if entry.total == 0:
|
|
|
|
if entry.total == 0:
|
|
|
|
continue
|
|
|
|
continue
|
|
|
@ -152,10 +156,10 @@ class UnpackOperation:
|
|
|
|
:return:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
"""
|
|
|
|
source_mount_path = tempfile.mkdtemp()
|
|
|
|
source_mount_path = tempfile.mkdtemp()
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
for entry in self.entries:
|
|
|
|
for entry in self.entries:
|
|
|
|
imgbasename = os.path.splitext(
|
|
|
|
imgbasename = os.path.splitext(os.path.basename(entry.source))[0]
|
|
|
|
os.path.basename(entry.source))[0]
|
|
|
|
|
|
|
|
imgmountdir = os.path.join(source_mount_path, imgbasename)
|
|
|
|
imgmountdir = os.path.join(source_mount_path, imgbasename)
|
|
|
|
os.mkdir(imgmountdir)
|
|
|
|
os.mkdir(imgmountdir)
|
|
|
|
|
|
|
|
|
|
|
@ -169,20 +173,19 @@ class UnpackOperation:
|
|
|
|
"Failed to find unsquashfs, make sure you have "
|
|
|
|
"Failed to find unsquashfs, make sure you have "
|
|
|
|
"the squashfs-tools package installed")
|
|
|
|
"the squashfs-tools package installed")
|
|
|
|
|
|
|
|
|
|
|
|
fslist = subprocess.check_output(["unsquashfs",
|
|
|
|
fslist = subprocess.check_output(["unsquashfs", "-l", entry.source])
|
|
|
|
"-l",
|
|
|
|
|
|
|
|
entry.source])
|
|
|
|
|
|
|
|
if entry.sourcefs == "ext4":
|
|
|
|
if entry.sourcefs == "ext4":
|
|
|
|
fslist = subprocess.check_output(["find",
|
|
|
|
fslist = subprocess.check_output(["find", imgmountdir, "-type", "f"])
|
|
|
|
imgmountdir,
|
|
|
|
|
|
|
|
"-type", "f"])
|
|
|
|
|
|
|
|
entry.total = len(fslist.splitlines())
|
|
|
|
entry.total = len(fslist.splitlines())
|
|
|
|
|
|
|
|
|
|
|
|
self.report_progress()
|
|
|
|
self.report_progress()
|
|
|
|
error_msg = self.unpack_image(entry, imgmountdir)
|
|
|
|
error_msg = self.unpack_image(entry, imgmountdir)
|
|
|
|
|
|
|
|
|
|
|
|
if error_msg:
|
|
|
|
if error_msg:
|
|
|
|
return ("Failed to unpack image {}".format(entry.source),
|
|
|
|
return "Failed to unpack image {}".format(entry.source), error_msg
|
|
|
|
error_msg)
|
|
|
|
|
|
|
|
return None
|
|
|
|
return None
|
|
|
|
finally:
|
|
|
|
finally:
|
|
|
|
shutil.rmtree(source_mount_path)
|
|
|
|
shutil.rmtree(source_mount_path)
|
|
|
@ -193,12 +196,7 @@ class UnpackOperation:
|
|
|
|
:param entry:
|
|
|
|
:param entry:
|
|
|
|
:param imgmountdir:
|
|
|
|
:param imgmountdir:
|
|
|
|
"""
|
|
|
|
"""
|
|
|
|
subprocess.check_call(["mount",
|
|
|
|
subprocess.check_call(["mount", entry.source, imgmountdir, "-t", entry.sourcefs, "-o", "loop"])
|
|
|
|
entry.source,
|
|
|
|
|
|
|
|
imgmountdir,
|
|
|
|
|
|
|
|
"-t",
|
|
|
|
|
|
|
|
entry.sourcefs,
|
|
|
|
|
|
|
|
"-o", "loop"])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def unpack_image(self, entry, imgmountdir):
|
|
|
|
def unpack_image(self, entry, imgmountdir):
|
|
|
|
""" Unpacks image.
|
|
|
|
""" Unpacks image.
|
|
|
@ -207,7 +205,6 @@ class UnpackOperation:
|
|
|
|
:param imgmountdir:
|
|
|
|
:param imgmountdir:
|
|
|
|
:return:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
def progress_cb(copied):
|
|
|
|
def progress_cb(copied):
|
|
|
|
""" Copies file to given destination target.
|
|
|
|
""" Copies file to given destination target.
|
|
|
|
|
|
|
|
|
|
|
@ -217,9 +214,7 @@ class UnpackOperation:
|
|
|
|
self.report_progress()
|
|
|
|
self.report_progress()
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
return file_copy(imgmountdir,
|
|
|
|
return file_copy(imgmountdir, entry.destination, progress_cb)
|
|
|
|
entry.destination,
|
|
|
|
|
|
|
|
progress_cb)
|
|
|
|
|
|
|
|
finally:
|
|
|
|
finally:
|
|
|
|
subprocess.check_call(["umount", "-l", imgmountdir])
|
|
|
|
subprocess.check_call(["umount", "-l", imgmountdir])
|
|
|
|
|
|
|
|
|
|
|
@ -245,14 +240,17 @@ def run():
|
|
|
|
PATH_PROCFS = '/proc/filesystems'
|
|
|
|
PATH_PROCFS = '/proc/filesystems'
|
|
|
|
|
|
|
|
|
|
|
|
root_mount_point = globalstorage.value("rootMountPoint")
|
|
|
|
root_mount_point = globalstorage.value("rootMountPoint")
|
|
|
|
|
|
|
|
|
|
|
|
if not root_mount_point:
|
|
|
|
if not root_mount_point:
|
|
|
|
return ("No mount point for root partition in globalstorage",
|
|
|
|
return ("No mount point for root partition in globalstorage",
|
|
|
|
"globalstorage does not contain a \"rootMountPoint\" key, "
|
|
|
|
"globalstorage does not contain a \"rootMountPoint\" key, "
|
|
|
|
"doing nothing")
|
|
|
|
"doing nothing")
|
|
|
|
|
|
|
|
|
|
|
|
if not os.path.exists(root_mount_point):
|
|
|
|
if not os.path.exists(root_mount_point):
|
|
|
|
return ("Bad mount point for root partition in globalstorage",
|
|
|
|
return ("Bad mount point for root partition in globalstorage",
|
|
|
|
"globalstorage[\"rootMountPoint\"] is \"{}\", which does not "
|
|
|
|
"globalstorage[\"rootMountPoint\"] is \"{}\", which does not "
|
|
|
|
"exist, doing nothing".format(root_mount_point))
|
|
|
|
"exist, doing nothing".format(root_mount_point))
|
|
|
|
|
|
|
|
|
|
|
|
unpack = list()
|
|
|
|
unpack = list()
|
|
|
|
|
|
|
|
|
|
|
|
for entry in job.configuration["unpack"]:
|
|
|
|
for entry in job.configuration["unpack"]:
|
|
|
@ -279,12 +277,13 @@ def run():
|
|
|
|
destination = os.path.abspath(root_mount_point + entry["destination"])
|
|
|
|
destination = os.path.abspath(root_mount_point + entry["destination"])
|
|
|
|
|
|
|
|
|
|
|
|
if not os.path.exists(source) or os.path.isdir(source):
|
|
|
|
if not os.path.exists(source) or os.path.isdir(source):
|
|
|
|
return ("Bad source", "source=\"{}\"".format(source))
|
|
|
|
return "Bad source", "source=\"{}\"".format(source)
|
|
|
|
|
|
|
|
|
|
|
|
if not os.path.isdir(destination):
|
|
|
|
if not os.path.isdir(destination):
|
|
|
|
return ("Bad destination",
|
|
|
|
return "Bad destination", "destination=\"{}\"".format(destination)
|
|
|
|
"destination=\"{}\"".format(destination))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unpack.append(UnpackEntry(source, sourcefs, destination))
|
|
|
|
unpack.append(UnpackEntry(source, sourcefs, destination))
|
|
|
|
|
|
|
|
|
|
|
|
unpackop = UnpackOperation(unpack)
|
|
|
|
unpackop = UnpackOperation(unpack)
|
|
|
|
|
|
|
|
|
|
|
|
return unpackop.run()
|
|
|
|
return unpackop.run()
|
|
|
|