Merge pull request #1622 from Chrysostomus/btrfs-subvol
[fstab][mount] Create and mount btrfs subvolumes in generalized manner
This commit is contained in:
commit
98d42719e1
@ -183,37 +183,19 @@ class FstabGenerator(object):
|
|||||||
print(FSTAB_HEADER, file=fstab_file)
|
print(FSTAB_HEADER, file=fstab_file)
|
||||||
|
|
||||||
for partition in self.partitions:
|
for partition in self.partitions:
|
||||||
# Special treatment for a btrfs root with @, @home and @swap
|
# Special treatment for a btrfs subvolumes
|
||||||
# subvolumes
|
|
||||||
if (partition["fs"] == "btrfs"
|
if (partition["fs"] == "btrfs"
|
||||||
and partition["mountPoint"] == "/"):
|
and partition["mountPoint"] == "/"):
|
||||||
output = subprocess.check_output(['btrfs',
|
# Subvolume list has been created in mount.conf and curated in mount module,
|
||||||
'subvolume',
|
# so all subvolumes here should be safe to add to fstab
|
||||||
'list',
|
btrfs_subvolumes = libcalamares.globalstorage.value("btrfsSubvolumes")
|
||||||
self.root_mount_point])
|
for s in btrfs_subvolumes:
|
||||||
output_lines = output.splitlines()
|
mount_entry = partition
|
||||||
for line in output_lines:
|
mount_entry["mountPoint"] = s["mountPoint"]
|
||||||
if line.endswith(b'path @'):
|
mount_entry["subvol"] = s["subvolume"]
|
||||||
root_entry = partition
|
dct = self.generate_fstab_line_info(mount_entry)
|
||||||
root_entry["subvol"] = "@"
|
|
||||||
dct = self.generate_fstab_line_info(root_entry)
|
|
||||||
if dct:
|
if dct:
|
||||||
self.print_fstab_line(dct, file=fstab_file)
|
self.print_fstab_line(dct, file=fstab_file)
|
||||||
elif line.endswith(b'path @home'):
|
|
||||||
home_entry = partition
|
|
||||||
home_entry["mountPoint"] = "/home"
|
|
||||||
home_entry["subvol"] = "@home"
|
|
||||||
dct = self.generate_fstab_line_info(home_entry)
|
|
||||||
if dct:
|
|
||||||
self.print_fstab_line(dct, file=fstab_file)
|
|
||||||
elif line.endswith(b'path @swap'):
|
|
||||||
swap_part_entry = partition
|
|
||||||
swap_part_entry["mountPoint"] = "/swap"
|
|
||||||
swap_part_entry["subvol"] = "@swap"
|
|
||||||
dct = self.generate_fstab_line_info(swap_part_entry)
|
|
||||||
if dct:
|
|
||||||
self.print_fstab_line(dct, file=fstab_file)
|
|
||||||
|
|
||||||
else:
|
else:
|
||||||
dct = self.generate_fstab_line_info(partition)
|
dct = self.generate_fstab_line_info(partition)
|
||||||
if dct:
|
if dct:
|
||||||
|
@ -32,9 +32,6 @@ def pretty_name():
|
|||||||
def mount_partition(root_mount_point, partition, partitions):
|
def mount_partition(root_mount_point, partition, partitions):
|
||||||
"""
|
"""
|
||||||
Do a single mount of @p partition inside @p root_mount_point.
|
Do a single mount of @p partition inside @p root_mount_point.
|
||||||
|
|
||||||
The @p partitions are used to handle btrfs special-cases:
|
|
||||||
then subvolumes are created for root and home.
|
|
||||||
"""
|
"""
|
||||||
# Create mount point with `+` rather than `os.path.join()` because
|
# Create mount point with `+` rather than `os.path.join()` because
|
||||||
# `partition["mountPoint"]` starts with a '/'.
|
# `partition["mountPoint"]` starts with a '/'.
|
||||||
@ -74,36 +71,29 @@ def mount_partition(root_mount_point, partition, partitions):
|
|||||||
partition.get("options", "")) != 0:
|
partition.get("options", "")) != 0:
|
||||||
libcalamares.utils.warning("Cannot mount {}".format(device))
|
libcalamares.utils.warning("Cannot mount {}".format(device))
|
||||||
|
|
||||||
# If the root partition is btrfs, we create a subvolume "@"
|
# Special handling for btrfs subvolumes. Create the subvolumes listed in mount.conf
|
||||||
# for the root mount point.
|
|
||||||
# If a separate /home partition isn't defined, we also create
|
|
||||||
# a subvolume "@home".
|
|
||||||
# If a swapfile is used, we also create a subvolume "@swap".
|
|
||||||
# Finally we remount all of the above on the correct paths.
|
|
||||||
if fstype == "btrfs" and partition["mountPoint"] == '/':
|
if fstype == "btrfs" and partition["mountPoint"] == '/':
|
||||||
has_home_mount_point = False
|
# Root has been mounted to btrfs volume -> create subvolumes from configuration
|
||||||
|
btrfs_subvolumes = libcalamares.job.configuration.get("btrfsSubvolumes") or []
|
||||||
|
subvolume_mountpoints = [d['mountPoint'] for d in btrfs_subvolumes]
|
||||||
|
# Check if listed mountpoints besides / are already present and don't create subvolume for those
|
||||||
for p in partitions:
|
for p in partitions:
|
||||||
if "mountPoint" not in p or not p["mountPoint"]:
|
if "mountPoint" not in p or not p["mountPoint"]:
|
||||||
continue
|
continue
|
||||||
if p["mountPoint"] == "/home":
|
if p["mountPoint"] in subvolume_mountpoints and p["mountPoint"] != '/':
|
||||||
has_home_mount_point = True
|
btrfs_subvolumes = [d for d in btrfs_subvolumes if d['mountPoint'] != p["mountPoint"]]
|
||||||
break
|
# Check if we need a subvolume for swap file
|
||||||
needs_swap_subvolume = False
|
|
||||||
swap_choice = global_storage.value( "partitionChoices" )
|
swap_choice = global_storage.value( "partitionChoices" )
|
||||||
if swap_choice:
|
if swap_choice:
|
||||||
swap_choice = swap_choice.get( "swap", None )
|
swap_choice = swap_choice.get( "swap", None )
|
||||||
if swap_choice and swap_choice == "file":
|
if swap_choice and swap_choice == "file":
|
||||||
needs_swap_subvolume = True
|
btrfs_subvolumes.append({'mountPoint': '/swap', 'subvolume': '/@swap'})
|
||||||
|
# Store created list in global storage so it can be used in the fstab module
|
||||||
|
libcalamares.globalstorage.insert("btrfsSubvolumes", btrfs_subvolumes)
|
||||||
|
# Create the subvolumes that are in the completed list
|
||||||
|
for s in btrfs_subvolumes:
|
||||||
subprocess.check_call(['btrfs', 'subvolume', 'create',
|
subprocess.check_call(['btrfs', 'subvolume', 'create',
|
||||||
root_mount_point + '/@'])
|
root_mount_point + s['subvolume']])
|
||||||
|
|
||||||
if not has_home_mount_point:
|
|
||||||
subprocess.check_call(['btrfs', 'subvolume', 'create',
|
|
||||||
root_mount_point + '/@home'])
|
|
||||||
if needs_swap_subvolume:
|
|
||||||
subprocess.check_call(['btrfs', 'subvolume', 'create',
|
|
||||||
root_mount_point + '/@swap'])
|
|
||||||
|
|
||||||
subprocess.check_call(["umount", "-v", root_mount_point])
|
subprocess.check_call(["umount", "-v", root_mount_point])
|
||||||
|
|
||||||
@ -112,24 +102,14 @@ def mount_partition(root_mount_point, partition, partitions):
|
|||||||
if "luksMapperName" in partition:
|
if "luksMapperName" in partition:
|
||||||
device = os.path.join("/dev/mapper", partition["luksMapperName"])
|
device = os.path.join("/dev/mapper", partition["luksMapperName"])
|
||||||
|
|
||||||
|
# Mount the subvolumes
|
||||||
|
for s in btrfs_subvolumes:
|
||||||
|
mount_option = "subvol={}".format(s['subvolume'])
|
||||||
|
subvolume_mountpoint = mount_point[:-1] + s['mountPoint']
|
||||||
if libcalamares.utils.mount(device,
|
if libcalamares.utils.mount(device,
|
||||||
mount_point,
|
subvolume_mountpoint,
|
||||||
fstype,
|
fstype,
|
||||||
",".join(["subvol=@", partition.get("options", "")])) != 0:
|
",".join([mount_option, partition.get("options", "")])) != 0:
|
||||||
libcalamares.utils.warning("Cannot mount {}".format(device))
|
|
||||||
|
|
||||||
if not has_home_mount_point:
|
|
||||||
if libcalamares.utils.mount(device,
|
|
||||||
root_mount_point + "/home",
|
|
||||||
fstype,
|
|
||||||
",".join(["subvol=@home", partition.get("options", "")])) != 0:
|
|
||||||
libcalamares.utils.warning("Cannot mount {}".format(device))
|
|
||||||
|
|
||||||
if needs_swap_subvolume:
|
|
||||||
if libcalamares.utils.mount(device,
|
|
||||||
root_mount_point + "/swap",
|
|
||||||
fstype,
|
|
||||||
",".join(["subvol=@swap", partition.get("options", "")])) != 0:
|
|
||||||
libcalamares.utils.warning("Cannot mount {}".format(device))
|
libcalamares.utils.warning("Cannot mount {}".format(device))
|
||||||
|
|
||||||
|
|
||||||
|
@ -40,3 +40,17 @@ extraMountsEfi:
|
|||||||
- device: efivarfs
|
- device: efivarfs
|
||||||
fs: efivarfs
|
fs: efivarfs
|
||||||
mountPoint: /sys/firmware/efi/efivars
|
mountPoint: /sys/firmware/efi/efivars
|
||||||
|
|
||||||
|
# Btrfs subvolumes to create if root filesystem is on btrfs volume.
|
||||||
|
# If mountpoint is mounted already to another partition, it is ignored.
|
||||||
|
# Separate subvolume for swapfile is handled separately and automatically.
|
||||||
|
|
||||||
|
btrfsSubvolumes:
|
||||||
|
- mountPoint: /
|
||||||
|
subvolume: /@
|
||||||
|
- mountPoint: /home
|
||||||
|
subvolume: /@home
|
||||||
|
- mountPoint: /var/cache
|
||||||
|
subvolume: /@cache
|
||||||
|
- mountPoint: /var/log
|
||||||
|
subvolume: /@log
|
@ -29,3 +29,12 @@ properties:
|
|||||||
mountPoint: { type: string }
|
mountPoint: { type: string }
|
||||||
options: { type: string }
|
options: { type: string }
|
||||||
required: [ device, mountPoint ]
|
required: [ device, mountPoint ]
|
||||||
|
btrfsSubvolumes:
|
||||||
|
type: array
|
||||||
|
items:
|
||||||
|
type: object
|
||||||
|
additionalProperties: false
|
||||||
|
properties:
|
||||||
|
mountPoint: { type: string }
|
||||||
|
subvolume: { type: string }
|
||||||
|
required: [ subvolume, mountPoint ]
|
||||||
|
Loading…
Reference in New Issue
Block a user