# Copyright 2016, Tresys Technology, LLC # # SPDX-License-Identifier: LGPL-2.1-only # # from contextlib import suppress from PyQt5 import QtCore, QtWidgets from setools import AnyTERule, TERuletype, TypeAttribute from setools.exception import RuleNotConditional, RuleUseError from . import modelroles from .table import SEToolsTableModel from ..details import objclass_detail, type_detail, type_or_attr_detail class TERuleTableModel(SEToolsTableModel[AnyTERule]): """A table-based model for TE rules.""" headers = ["Rule Type", "Source", "Target", "Object Class", "Permissions/Default Type", "Conditional Expression", "Conditional Branch"] def data(self, index: QtCore.QModelIndex, role: int = QtCore.Qt.ItemDataRole.DisplayRole): if not self.item_list or not index.isValid(): return None row = index.row() col = index.column() rule = self.item_list[row] if role == QtCore.Qt.ItemDataRole.DisplayRole: if col == 0: return rule.ruletype.name elif col == 1: return rule.source.name elif col == 2: return rule.target.name elif col == 3: return rule.tclass.name elif col == 4: try: if rule.extended: return f"{rule.xperm_type}: {rule.perms:,}" # type: ignore else: return ", ".join(sorted(rule.perms)) # type: ignore except RuleUseError: return rule.default.name # type: ignore elif col == 5: with suppress(RuleNotConditional): return str(rule.conditional) elif col == 6: with suppress(RuleNotConditional): return str(rule.conditional_block) return None elif role == modelroles.ContextMenuRole: if col == 1: a = QtWidgets.QAction(f"Properties of {rule.source}") a.triggered.connect(lambda x: type_or_attr_detail(rule.source)) return (a, ) elif col == 2: a = QtWidgets.QAction(f"Properties of {rule.target}") a.triggered.connect(lambda x: type_or_attr_detail(rule.target)) return (a, ) elif col == 3: a = QtWidgets.QAction(f"Properties of {rule.tclass}") a.triggered.connect(lambda x: objclass_detail(rule.tclass)) return (a, ) elif col == 4: with suppress(RuleUseError): a = QtWidgets.QAction(f"Properties of {rule.default}") a.triggered.connect(lambda x: type_detail(rule.default)) return (a, ) return () elif role == QtCore.Qt.ItemDataRole.ToolTipRole: if col in (1, 2): if col == 1: obj = rule.source else: obj = rule.target if isinstance(obj, TypeAttribute): n_types = len(obj) if n_types == 0: return f"{obj.name} is an empty attribute." elif n_types > 5: return f"{obj.name} is an attribute consisting of {n_types} types." else: return f"{obj.name} is an attribute consisting of: " \ f"{', '.join(t.name for t in obj.expand())}" else: return f"{obj.name} is a type." return None elif role == QtCore.Qt.ItemDataRole.WhatsThisRole: if col == 0: column_whatsthis = \ f"""
The Rule Type column is the type of the rule; it is one of:
This is the source type or type attribute (subject) in the rule.
" elif col == 2: column_whatsthis = \ "This is the target type or type attribute (object) in the rule.
" elif col == 3: column_whatsthis = "This is the object class of the rule.
" elif col == 4: column_whatsthis = \ """Permissions/Default Type: The value of this depends on the rule type:
This is the conditional expression that enables/disables this rule. If this is blank, the rule is unconditional.
""" elif col == 6: column_whatsthis = \ """This contains the conditional branch that that rule resides in. "True" means the rule is enabled when the conditional expression is true; also known as the "if" block. "False" means the rule is enabled when the conditional expression is false; also known as the "else" block. If this is blank, the rule is unconditional.
""" return \ f"""Table Representation of Type Enforcement Rules
Each part of the rule is represented as a column in the table.
{column_whatsthis} """ return super().data(index, role)