Source code for insights.parsers.dmsetup

"""
dmsetup commands - Command ``dmsetup``
======================================

Parsers for parsing and extracting data from output of commands related to
``dmsetup``.
Parsers contained in this module are:

DmsetupInfo - command ``dmsetup info -C``
-----------------------------------------

DmsetupStatus - command ``dmsetup status``
------------------------------------------

"""
from collections import namedtuple

from insights.core import CommandParser
from insights.core.exceptions import ParseException
from insights.core.plugins import parser
from insights.parsers import parse_delimited_table
from insights.specs import Specs


SetupInfo = namedtuple('SetupInfo', [
    'name', 'major', 'minor', 'open', 'segments', 'events',
    'live_table', 'inactive_table', 'suspended', 'readonly', 'uuid']
)
""" Data structure to represent dmsetup information """

SetupStatus = namedtuple('SetupStatus', [
    'device_name', 'start_sector', 'num_sectors', 'target_type',
    'target_args', 'parsed_args']
)
""" Data structure to represent dmsetup status """


[docs] @parser(Specs.dmsetup_info) class DmsetupInfo(CommandParser): """ ``dmsetup info -C`` command output Example input:: Name Maj Min Stat Open Targ Event UUID VG00-tmp 253 8 L--w 1 1 0 LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTax6lLmBji2ueSbX49gxcV76M29cmukQiw4 VG00-home 253 3 L--w 1 1 0 LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTaxCqXOnbGe2zjhX923dFiIdl1oi7mO9tXp VG00-var 253 6 L--w 1 2 0 LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTaxicvyvt67113nTb8vMlGfgdEjDx0LKT2O VG00-swap 253 1 L--w 2 1 0 LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTax3Ll2XhOYZkylx1CjOQi7G4yHgrIOsyqG VG00-root 253 0 L--w 1 1 0 LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTaxKpnAKYhrYMYMNMwjegkW965bUgtJFTRY VG00-var_log_audit 253 5 L--w 1 1 0 LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTaxwQ8R0XWJRm86QX3befq1cHRy47Von6ZW Example data structure produced:: data = [ { 'Stat': 'L--w', 'Name': 'VG00-tmp', 'Min': '8', 'Targ': '1', 'Maj': '253', 'Open': '1', 'Event': '0', 'UUID': 'LVM-gy9uAwD7LuTIApplr2sogbOx5iS0FTax6lLmBji2ueSbX49gxcV76M29cmukQiw4' },... ] Attributes: data (list): List of devices found, in order names (list): Device names, in order found uuids (list): UUID by_name (dict): Access to each device by devicename by_uuid (dict): Access to each device by uuid info (list): List of devices found, in order using SetupInfo structure Example: >>> len(setup_info) 6 >>> setup_info.names[0] 'VG00-tmp' >>> setup_info[1]['Maj'] '253' >>> setup_info[1]['Stat'] 'L--w' >>> setup_info.info[-1].name 'VG00-var_log_audit' >>> setup_info.info[-1].major 253 >>> setup_info.info[-1].live_table True >>> setup_info.info[-1].readonly False """
[docs] def parse_content(self, content): self.data = parse_delimited_table(content) self.names = [dm['Name'] for dm in self.data if 'Name' in dm] self.by_name = dict((dm['Name'], dm) for dm in self.data if 'Name' in dm) self.uuids = [dm['UUID'] for dm in self.data if 'UUID' in dm] self.by_uuid = dict((dm['UUID'], dm) for dm in self.data if 'UUID' in dm) self.info = [] for dm in self.data: self.info.append(SetupInfo( name=dm.get('Name'), major=int(dm.get('Maj')) if 'Maj' in dm and dm.get('Maj').isdigit() else None, minor=int(dm.get('Min')) if 'Min' in dm and dm.get('Min').isdigit() else None, open=int(dm.get('Open')) if 'Open' in dm and dm.get('Open').isdigit() else None, segments=int(dm.get('Targ')) if 'Targ' in dm and dm.get('Targ').isdigit() else None, events=int(dm.get('Event')) if 'Event' in dm and dm.get('Event').isdigit() else None, live_table=dm.get('Stat', '----')[0] == 'L', inactive_table=dm.get('Stat', '----')[1] == 'I', suspended=dm.get('Stat', '----')[2] == 's', readonly=dm.get('Stat', '----')[3] == 'r', uuid=dm.get('UUID') ))
def __len__(self): """ The length of the devices list """ return len(self.data) def __iter__(self): """ Iterate through the devices list """ for dm in self.data: yield dm def __getitem__(self, idx): """ Fetch a device by index in devices list """ return self.data[idx]
[docs] @parser(Specs.dmsetup_status) class DmsetupStatus(CommandParser, list): """ ``dmsetup status -C`` command output Example input:: rootvg-tanlv: 0 6291456 linear rootvg-ssnap: 0 16384000 snapshot 1560768/5120000 6088 rootvg-optvolapp: 0 8192000 snapshot-origin docker-253:10-1234567-0df13579: 0 20971520 thin 1922048 20971519 docker-253:10-4254621-0496628a: 0 20971520 thin 1951744 20971519 docker-253:10-4254621-d392682f: 0 20971520 thin 7106560 20971519 rootvg-docker--pool: 0 129548288 thin-pool 1 20/49152 38/126512 - rw no_discard_passdown queue_if_no_space - rootvg-tmpvol: 0 2048000 linear rootvg-varvol: 0 18874368 snapshot Invalid rootvg-optvol: 0 8192000 snapshot 616408/5120000 2408 rootvg-varvol-cow: 0 5120000 linear appsvg-lvapps_docker: 0 104857600 thin-pool 441 697/2048 20663/102400 - rw no_discard_passdown queue_if_no_space - Example data structure produced:: [ SetupStatus( device_name='rootvg-tanlv', start_sector='0', num_sectors='6291456', target_type='linear', target_args=None, parsed_args=None, ), ... ] Attributes: names (list): Device names, in order found by_name (dict): Access to each device by devicename unparseable_lines (list): Unparseable raw lines Example: >>> len(dmsetup_status) 12 >>> dmsetup_status.names[0] 'rootvg-tanlv' >>> dmsetup_status[1].target_type 'snapshot' >>> dmsetup_status[1].start_sector '0' >>> len(dmsetup_status.by_name) 12 >>> dmsetup_status[-1].parsed_args['used_metadata_blocks'] '697' >>> dmsetup_status[-1].parsed_args['total_metadata_blocks'] '2048' >>> dmsetup_status[-1].parsed_args['opts'] ['rw', 'no_discard_passdown', 'queue_if_no_space', '-'] """
[docs] def parse_content(self, content): self.unparseable_lines = [] if content[0].lower() == "no devices found": return for line in content: _device_name, _device_info_str = line.rsplit(':', 1) device_name = _device_name.strip() device_info_spl = _device_info_str.strip().split(' ', 3) if len(device_info_spl) < 3: self.unparseable_lines.append(line) continue target_type = device_info_spl[2] target_args = device_info_spl[3] if len(device_info_spl) == 4 else None parsed_args = None if target_args: try: parsed_args = self._parse_target_args(target_type, target_args) except ParseException: self.unparseable_lines.append(line) self.append(SetupStatus( device_name=device_name, start_sector=device_info_spl[0], num_sectors=device_info_spl[1], target_type=target_type, target_args=target_args, parsed_args=parsed_args, ))
@property def names(self): return [dm[0] for dm in self] @property def by_name(self): return dict((dm[0], dm) for dm in self) def _parse_target_args(self, target_type, target_args): pars_func_name = '_parse_target_args_' + target_type.replace('-', '_') pars_func = getattr(self, pars_func_name, None) return pars_func(target_args) if pars_func else None def _parse_target_args_thin_pool(self, target_args): """ Format: <transaction_id> <used_metadata_blocks>/<total_metadata_blocks> <used_data_blocks>/<total_data_blocks> <held_metadata_root> ro|rw|out_of_data_space [no_]discard_passdown [error|queue]_if_no_space needs_check|- metadata_low_watermark Refer to https://www.kernel.org/doc/Documentation/device-mapper/thin-provisioning.txt . """ args = target_args.split() if len(args) < 8 or '/' not in args[1] or '/' not in args[2]: raise ParseException("Invalid thin_pool target_args: {0}".format(target_args)) parsed_args = {} parsed_args['transaction_id'] = args[0] parsed_args['used_metadata_blocks'], parsed_args['total_metadata_blocks'] = args[1].split('/', 1) parsed_args['used_data_blocks'], parsed_args['total_data_blocks'] = args[2].split('/', 1) parsed_args['held_metadata_root'] = args[3] parsed_args['opts'] = args[4:8] parsed_args['metadata_low_watermark'] = args[8] if len(args) > 8 else None return parsed_args def _parse_target_args_thin(self, target_args): """ Format: <nr_mapped_sectors> <highest_mapped_sector> Refer to https://www.kernel.org/doc/Documentation/device-mapper/thin-provisioning.txt . """ args = target_args.split() if len(args) < 2: raise ParseException("Invalid thin target_args: {0}".format(target_args)) parsed_args = {} parsed_args['nr_mapped_sectors'] = args[0] parsed_args['highest_mapped_sector'] = args[1] return parsed_args def _parse_target_args_snapshot(self, target_args): """ Format: <sectors_allocated>/<total_sectors> <metadata_sectors> Refer to https://www.kernel.org/doc/Documentation/device-mapper/snapshot.txt . """ args = target_args.split() if len(args) < 2 or '/' not in args[0]: raise ParseException("Invalid snapshot target_args: {0}".format(target_args)) parsed_args = {} parsed_args['sectors_allocated'], parsed_args['total_sectors'] = args[0].split('/', 1) parsed_args['metadata_sectors'] = args[1] return parsed_args