Skip to content
Snippets Groups Projects
Commit 9517bf29 authored by Nikos Skalkotos's avatar Nikos Skalkotos
Browse files

Fix the image partitions in bundle_volume

Check the file system usage and accordingly reduce the size of the
last partition in the image.
parent 6523456e
No related branches found
No related tags found
No related merge requests found
...@@ -35,6 +35,7 @@ import os ...@@ -35,6 +35,7 @@ import os
import re import re
import parted import parted
import uuid import uuid
from collections import namedtuple
from image_creator.util import get_command from image_creator.util import get_command
from image_creator.util import FatalError from image_creator.util import FatalError
...@@ -43,46 +44,66 @@ findfs = get_command('findfs') ...@@ -43,46 +44,66 @@ findfs = get_command('findfs')
truncate = get_command('truncate') truncate = get_command('truncate')
dd = get_command('dd') dd = get_command('dd')
def get_root_partition():
if not os.path.isfile('/etc/fstab'):
raise FatalError("Unable to open `/etc/fstab'. File is missing.")
with open('/etc/fstab') as fstab: def fstable(f):
for line in iter(fstab): if not os.path.isfile(f):
raise FatalError("Unable to open: `%s'. File is missing." % f)
Entry = namedtuple('Entry', 'dev mpoint fs opts freq passno')
with open(f) as table:
for line in iter(table):
entry = line.split('#')[0].strip().split() entry = line.split('#')[0].strip().split()
if len(entry) != 6: if len(entry) != 6:
continue continue
yield Entry(*entry)
if entry[1] == "/":
return entry[0]
raise FatalError("Unable to find root device in /etc/fstab") def get_root_partition():
for entry in fstable('/etc/fstab'):
if entry.mpoint == '/':
return entry.dev
def mnt_mounted(): raise FatalError("Unable to find root device in /etc/fstab")
if not os.path.isfile('/etc/mtab'):
raise FatalError("Unable to open `/etc/fstab'. File is missing.")
with open('/etc/mtab') as mtab:
for line in iter(mtab):
entry = line.split('#')[0].strip().split()
if len(entry) != 6:
continue
if entry[1] == '/mnt':
return True
def is_mpoint(path):
for entry in fstable('/proc/mounts'):
if entry.mpoint == path:
return True
return False return False
def part_to_dev(part): def mpoint(device):
return re.split('[0-9]', part)[0] for entry in fstable('/proc/mounts'):
if not entry.dev.startswith('/'):
continue
if os.path.realpath(entry.dev) == os.path.realpath(device):
return entry.mpoint
def part_to_num(part): return ""
return re.split('[^0-9]+', part)[-1]
def bundle_volume(out):
if mnt_mounted(): def create_EBRs(src, dest):
# The Extended boot records precede the logical partitions they describe
extended = src.getExtendedPartition()
if not extended:
return
logical = src.getLogicalPartitions()
start = extended.geometry.start
for l in range(len(logical)):
end = l.geometry.start - 1
dd('if=%s' % src.path, 'of=%s' % dest, 'count=%d' % (end - start + 1),
'conv=notrunc', 'seek=%d' % start, 'skip=%d' % start)
start = l.geometry.end + 1
def bundle_volume(out, meta):
if is_mpoint('/mnt'):
raise FatalError('The directory /mnt where the image will be hosted' raise FatalError('The directory /mnt where the image will be hosted'
'is mounted. Please unmount it and start over again.') 'is mounted. Please unmount it and start over again.')
...@@ -97,24 +118,98 @@ def bundle_volume(out): ...@@ -97,24 +118,98 @@ def bundle_volume(out):
if not re.match('/dev/[hsv]d[a-z][1-9]*$', root_part): if not re.match('/dev/[hsv]d[a-z][1-9]*$', root_part):
raise FatalError("Don't know how to handle root device: %s" % root_dev) raise FatalError("Don't know how to handle root device: %s" % root_dev)
device = parted.Device(part_to_dev(root_part)) part_to_dev = lambda p: re.split('[0-9]', p)[0]
root_dev = part_to_dev(root_part)
out.success('%s' % root_dev)
src_dev = parted.Device(root_dev)
image = '/mnt/%s.diskdump' % uuid.uuid4().hex img = '/mnt/%s.diskdump' % uuid.uuid4().hex
disk_size = src_dev.getLength() * src_dev.sectorSize
# Create sparse file to host the image # Create sparse file to host the image
truncate("-s", "%d" % (device.getLength() * device.sectorSize), image) truncate("-s", "%d" % disk_size, img)
disk = parted.Disk(device) src_disk = parted.Disk(src_dev)
if disk.type != 'msdos': if src_disk.type != 'msdos':
raise FatalError('For now we can only handle msdos partition tables') raise FatalError('For now we can only handle msdos partition tables')
# Copy the MBR and the space between the MBR and the first partition. # Copy the MBR and the space between the MBR and the first partition.
# In Grub version 1 Stage 1.5 is located there. # In Grub version 1 Stage 1.5 is located there.
first_sector = disk.getPrimaryPartitions()[0].geometry.start first_sector = src_disk.getPrimaryPartitions()[0].geometry.start
dd('if=%s' % device.path, 'of=%s' % image, 'bs=%d' % device.sectorSize, dd('if=%s' % src_dev.path, 'of=%s' % img, 'bs=%d' % src_dev.sectorSize,
'count=%d' % first_sector, 'conv=notrunc') 'count=%d' % first_sector, 'conv=notrunc')
return image # Create the Extended boot records (EBRs) in the image
create_EBRs(src_disk, img)
img_dev = parted.Device(img)
img_disk = parted.Disk(img_dev)
Partition = namedtuple('Partition', 'num start end type fs mpoint')
partitions = []
for p in src_disk.partitions:
g = p.geometry
f = p.fileSystem
partitions.append(Partition(p.number, g.start, g.end, p.type,
f.type if f is not None else '', mpoint(p.path)))
last = partitions[-1]
new_end = src_dev.getLength()
if last.fs == 'linux-swap(v1)':
size = (last.end - last.start + 1) * src_dev.sectorSize
MB = 2 ** 20
meta['SWAP'] = "%d:%s" % (last.num, ((size + MB - 1) // MB))
img_disk.deletePartition(img_disk.getPartitionBySector(last.start))
if last.type == parted.PARTITION_LOGICAL and last.num == 5:
img_disk.deletePartition(img_disk.getExtendedPartition())
partitions.remove(last)
last = partitions[-1]
# Leave 2048 blocks at the end
new_end = last.end + 2048
if last.mpoint:
stat = statvfs(last.mpoint)
occupied_blocks = stat.f_blocks - stat.f_bavail
new_size = (occupied * stat.f_frsize) // src_dev.sectorSize
# Add 10% just to be on the safe side
last.end = last.start + (new_size * 11) // 10
# Alighn to 2048
last.end = ((last.end + 2047) // 2048) * 2048
# Leave 2048 blocks at the end.
new_end = new_size + 2048
img_disk.setPartitionGeometry(
img_disk.getPartitionBySector(last.start),
parted.Constraint(device=img_dev), start=last.star, end=last.end)
if last.type == parted.PARTITION_LOGICAL:
# Fix the extended partition
ext = disk.getExtendedPartition()
img_disk.setPartitionGeometry(ext,
parted.Constraint(device=img_dev), ext.geometry.start,
end=last.end)
# Check if we have the available space on the filesystem hosting /mnt
# for the image.
out.output("Examining available space in /mnt ... ", False)
stat = os.statvfs('/mnt')
image_size = (new_end + 1) * src_dev.sectorSize
available = stat.f_bavail * stat.f_frsize
if available <= image_size:
raise FatalError('Not enough space in /mnt to host the image')
out.success("sufficient")
return img
# vim: set sta sts=4 shiftwidth=4 sw=4 et ai : # vim: set sta sts=4 shiftwidth=4 sw=4 et ai :
...@@ -67,6 +67,7 @@ class Disk(object): ...@@ -67,6 +67,7 @@ class Disk(object):
self._devices = [] self._devices = []
self.source = source self.source = source
self.out = output self.out = output
self.meta = {}
def _add_cleanup(self, job, *args): def _add_cleanup(self, job, *args):
self._cleanup_jobs.append((job, args)) self._cleanup_jobs.append((job, args))
...@@ -79,7 +80,7 @@ class Disk(object): ...@@ -79,7 +80,7 @@ class Disk(object):
def _dir_to_disk(self): def _dir_to_disk(self):
if self.source == '/': if self.source == '/':
return bundle_volume(self.out) return bundle_volume(self.out, self.meta)
raise FatalError("Using a directory as media source is supported") raise FatalError("Using a directory as media source is supported")
def cleanup(self): def cleanup(self):
...@@ -165,16 +166,16 @@ class DiskDevice(object): ...@@ -165,16 +166,16 @@ class DiskDevice(object):
as created by the device-mapper. as created by the device-mapper.
""" """
def __init__(self, device, output, bootable=True): def __init__(self, device, output, bootable=True, meta={}):
"""Create a new DiskDevice.""" """Create a new DiskDevice."""
self.real_device = device self.real_device = device
self.out = output self.out = output
self.bootable = bootable self.bootable = bootable
self.meta = meta
self.progress_bar = None self.progress_bar = None
self.guestfs_device = None self.guestfs_device = None
self.size = 0 self.size = 0
self.meta = {}
self.g = guestfs.GuestFS() self.g = guestfs.GuestFS()
self.g.add_drive_opts(self.real_device, readonly=0) self.g.add_drive_opts(self.real_device, readonly=0)
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment