2019-11-01 02:54:51 -04:00
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
##
|
|
|
|
import psutil
|
|
|
|
##
|
|
|
|
import aif.disk.block_fallback as block
|
|
|
|
import aif.disk.luks_fallback as luks
|
|
|
|
import aif.disk.lvm_fallback as lvm
|
|
|
|
import aif.disk.mdadm_fallback as mdadm
|
2019-11-06 02:21:04 -05:00
|
|
|
import aif.utils
|
2019-11-01 02:54:51 -04:00
|
|
|
|
2019-11-06 02:21:04 -05:00
|
|
|
|
|
|
|
FS_FSTYPES = aif.utils.kernelFilesystems()
|
2019-11-01 02:54:51 -04:00
|
|
|
|
|
|
|
|
|
|
|
class FS(object):
|
|
|
|
def __init__(self, fs_xml, sourceobj):
|
|
|
|
self.xml = fs_xml
|
2019-11-06 02:21:04 -05:00
|
|
|
if not isinstance(sourceobj, (block.Disk,
|
|
|
|
block.Partition,
|
|
|
|
luks.LUKS,
|
|
|
|
lvm.LV,
|
|
|
|
mdadm.Array)):
|
2019-11-01 02:54:51 -04:00
|
|
|
raise ValueError(('sourceobj must be of type '
|
|
|
|
'aif.disk.block.Partition, '
|
|
|
|
'aif.disk.luks.LUKS, '
|
|
|
|
'aif.disk.lvm.LV, or'
|
|
|
|
'aif.disk.mdadm.Array'))
|
2019-11-06 02:21:04 -05:00
|
|
|
self.id = self.xml.attrib['id']
|
2019-11-01 02:54:51 -04:00
|
|
|
self.source = sourceobj
|
|
|
|
self.devpath = sourceobj.devpath
|
|
|
|
self.formatted = False
|
|
|
|
self.fstype = self.xml.attrib.get('type')
|
|
|
|
|
|
|
|
def format(self):
|
|
|
|
if self.formatted:
|
|
|
|
return ()
|
|
|
|
# This is a safeguard. We do *not* want to high-format a disk that is mounted.
|
2019-11-06 02:21:04 -05:00
|
|
|
aif.utils.checkMounted(self.devpath)
|
2019-11-01 02:54:51 -04:00
|
|
|
# TODO! Logging
|
|
|
|
cmd = ['mkfs',
|
|
|
|
'-t', self.fstype]
|
|
|
|
for o in self.xml.findall('opt'):
|
|
|
|
cmd.append(o.attrib['name'])
|
|
|
|
if o.text:
|
|
|
|
cmd.append(o.text)
|
|
|
|
cmd.append(self.devpath)
|
|
|
|
subprocess.run(cmd)
|
2019-11-06 02:21:04 -05:00
|
|
|
self.formatted = True
|
2019-12-11 04:33:15 -05:00
|
|
|
return(None)
|
2019-11-06 02:21:04 -05:00
|
|
|
|
|
|
|
|
|
|
|
class Mount(object):
|
|
|
|
def __init__(self, mount_xml, fsobj):
|
|
|
|
self.xml = mount_xml
|
|
|
|
self.id = self.xml.attrib['id']
|
|
|
|
if not isinstance(fsobj, FS):
|
|
|
|
raise ValueError('partobj must be of type aif.disk.filesystem.FS')
|
|
|
|
self.id = self.xml.attrib['id']
|
|
|
|
self.fs = fsobj
|
|
|
|
self.source = self.fs.devpath
|
|
|
|
self.target = os.path.realpath(self.xml.attrib['target'])
|
|
|
|
self.opts = {}
|
|
|
|
for o in self.xml.findall('opt'):
|
|
|
|
self.opts[o.attrib['name']] = o.text
|
|
|
|
self.mounted = False
|
|
|
|
|
|
|
|
def _parseOpts(self):
|
|
|
|
opts = []
|
|
|
|
for k, v in self.opts.items():
|
|
|
|
if v and v is not True: # Python's boolean determination is weird sometimes.
|
|
|
|
opts.append('{0}={1}'.format(k, v))
|
|
|
|
else:
|
|
|
|
opts.append(k)
|
|
|
|
return(opts)
|
|
|
|
|
|
|
|
def mount(self):
|
|
|
|
if self.mounted:
|
2019-12-11 04:33:15 -05:00
|
|
|
return(None)
|
2019-11-06 02:21:04 -05:00
|
|
|
os.makedirs(self.target, exist_ok = True)
|
|
|
|
opts = self._parseOpts()
|
|
|
|
# TODO: logging
|
|
|
|
cmd = ['/usr/bin/mount',
|
|
|
|
'--types', self.fs.fstype]
|
|
|
|
if opts:
|
|
|
|
cmd.extend(['--options', ','.join(opts)])
|
|
|
|
cmd.extend([self.source, self.target])
|
|
|
|
subprocess.run(cmd)
|
|
|
|
self.mounted = True
|
2019-12-11 04:33:15 -05:00
|
|
|
return(None)
|
2019-11-06 02:21:04 -05:00
|
|
|
|
|
|
|
def unmount(self, lazy = False, force = False):
|
|
|
|
self.updateMount()
|
|
|
|
if not self.mounted and not force:
|
2019-12-11 04:33:15 -05:00
|
|
|
return(None)
|
2019-11-06 02:21:04 -05:00
|
|
|
# TODO: logging
|
|
|
|
cmd = ['/usr/bin/umount']
|
|
|
|
if lazy:
|
|
|
|
cmd.append('--lazy')
|
|
|
|
if force:
|
|
|
|
cmd.append('--force')
|
|
|
|
cmd.append(self.target)
|
|
|
|
subprocess.run(cmd)
|
|
|
|
self.mounted = False
|
2019-12-11 04:33:15 -05:00
|
|
|
return(None)
|
2019-11-06 02:21:04 -05:00
|
|
|
|
|
|
|
def updateMount(self):
|
|
|
|
if self.source in [p.device for p in psutil.disk_partitions(all = True)]:
|
|
|
|
self.mounted = True
|
|
|
|
else:
|
|
|
|
self.mounted = False
|
2019-12-11 04:33:15 -05:00
|
|
|
return(None)
|