qubes-installer-qubes-os/anaconda/pyanaconda/packaging/livepayload.py

142 lines
5.1 KiB
Python
Raw Normal View History

# livepayload.py
# Live media software payload management.
#
# Copyright (C) 2012 Red Hat, Inc.
#
# This copyrighted material is made available to anyone wishing to use,
# modify, copy, or redistribute it subject to the terms and conditions of
# the GNU General Public License v.2, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY expressed or implied, including the implied warranties 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, write to the
# Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
# 02110-1301, USA. Any Red Hat trademarks that are incorporated in the
# source code or documentation are not subject to the GNU General Public
# License and may only be used or replicated with the express permission of
# Red Hat, Inc.
#
# Red Hat Author(s): David Lehman <dlehman@redhat.com>
#
"""
TODO
- error handling!!!
- document all methods
- LiveImagePayload
- register the live image, either via self.data.method or in setup
using storage
"""
import os
import stat
from time import sleep
from threading import Lock
from pyanaconda import isys
from . import *
from pyanaconda.constants import *
from pyanaconda.flags import flags
from pyanaconda import iutil
import logging
log = logging.getLogger("anaconda")
from pyanaconda.errors import *
from pyanaconda import progress
from pyanaconda.storage.size import Size
from pyanaconda.threads import threadMgr, AnacondaThread
import gettext
_ = lambda x: gettext.ldgettext("anaconda", x)
class LiveImagePayload(ImagePayload):
""" A LivePayload copies the source image onto the target system. """
def setup(self, storage):
super(LiveImagePayload, self).setup(storage)
# Mount the live device and copy from it instead of the overlay at /
osimg = storage.devicetree.getDeviceByPath(self.data.method.partition)
if not stat.S_ISBLK(os.stat(osimg.path)[stat.ST_MODE]):
exn = PayloadSetupError("%s is not a valid block device" % (self.data.method.partition,))
if errorHandler.cb(exn) == ERROR_RAISE:
raise exn
isys.mount(osimg.path, INSTALL_TREE, fstype="auto", readOnly=True)
def preInstall(self, packages=None, groups=None):
""" Perform pre-installation tasks. """
super(LiveImagePayload, self).preInstall(packages=packages, groups=groups)
progress.send_message(_("Installing software") + (" %d%%") % (0,))
def progress(self):
"""Monitor the amount of disk space used on the target and source and
update the hub's progress bar.
"""
source = os.statvfs(INSTALL_TREE)
source_size = source.f_frsize * (source.f_blocks - source.f_bfree)
mountpoints = self.storage.mountpoints.copy()
while self.pct < 100:
dest_size = 0
for mnt in mountpoints:
mnt_stat = os.statvfs(ROOT_PATH+mnt)
dest_size += mnt_stat.f_frsize * (mnt_stat.f_blocks - mnt_stat.f_bfree)
with self.pct_lock:
self.pct = int(100 * dest_size / source_size)
progress.send_message(_("Installing software") + (" %d%%") % (min(100,self.pct),))
sleep(0.777)
def install(self):
""" Install the payload. """
self.pct_lock = Lock()
self.pct = 0
threadMgr.add(AnacondaThread(name="AnaLiveProgressThread",
target=self.progress))
cmd = "rsync"
# preserve: permissions, owners, groups, ACL's, xattrs, times,
# symlinks, hardlinks
# go recursively, include devices and special files, don't cross
# file system boundaries
args = ["-pogAXtlHrDx", "--exclude", "/dev/", "--exclude", "/proc/",
"--exclude", "/sys/", INSTALL_TREE+"/", ROOT_PATH]
try:
rc = iutil.execWithRedirect(cmd, args,
stderr="/dev/tty5", stdout="/dev/tty5")
except (OSError, RuntimeError) as e:
msg = None
err = str(e)
log.error(err)
else:
err = None
msg = "%s exited with code %d" % (cmd, rc)
log.info(msg)
if err or rc == 12:
exn = PayloadInstallError(err or msg)
if errorHandler.cb(exn) == ERROR_RAISE:
raise exn
# Wait for progress thread to finish
with self.pct_lock:
self.pct = 100
progressThread = threadMgr.get("AnaLiveProgressThread")
if progressThread:
progressThread.join()
def postInstall(self):
""" Perform post-installation tasks. """
progress.send_message(_("Performing post-install setup tasks"))
isys.umount(INSTALL_TREE, removeDir=True)
super(LiveImagePayload, self).postInstall()
self._recreateInitrds()
@property
def spaceRequired(self):
return Size(bytes=iutil.getDirSize("/")*1024)