Improve MLS object representation.

Focused on completeness of the objects, particularly for comparing
levels.

Also leveraged qpol_level_t (level declaration) for MLSSensitivity since
qpol doesn't have a sensitivity object, and there is a 1:1 correspondence
between sensitivity declarations and level statements.
This commit is contained in:
Chris PeBenito 2015-02-20 10:35:23 -05:00
parent cf601f08fd
commit 880582fe73
5 changed files with 165 additions and 36 deletions

View File

@ -1,4 +1,4 @@
# Copyright 2014, Tresys Technology, LLC # Copyright 2014-2015, Tresys Technology, LLC
# #
# This file is part of SETools. # This file is part of SETools.
# #
@ -317,6 +317,12 @@ class SELinuxPolicy(object):
# where a class has no default_* settings. # where a class has no default_* settings.
pass pass
def levels(self):
"""Generator which yields all level declarations."""
for level in self.policy.level_iter():
yield mls.level_decl_factory(self.policy, level)
def types(self): def types(self):
"""Generator which yields all types.""" """Generator which yields all types."""

View File

@ -1,4 +1,4 @@
# Copyright 2014, Tresys Technology, LLC # Copyright 2014-2015, Tresys Technology, LLC
# #
# This file is part of SETools. # This file is part of SETools.
# #
@ -39,7 +39,7 @@ class Context(symbol.PolicySymbol):
def __str__(self): def __str__(self):
try: try:
return "{0.user}:{0.role}:{0.type_}:{0.mls}".format(self) return "{0.user}:{0.role}:{0.type_}:{0.range_}".format(self)
except mls.MLSDisabled: except mls.MLSDisabled:
return "{0.user}:{0.role}:{0.type_}".format(self) return "{0.user}:{0.role}:{0.type_}".format(self)
@ -59,8 +59,8 @@ class Context(symbol.PolicySymbol):
return typeattr.type_factory(self.policy, self.qpol_symbol.type_(self.policy)) return typeattr.type_factory(self.policy, self.qpol_symbol.type_(self.policy))
@property @property
def mls(self): def range_(self):
"""The MLS portion (range) of the context.""" """The MLS range of the context."""
# without this check, qpol will segfault on MLS-disabled policies # without this check, qpol will segfault on MLS-disabled policies
if self.policy.capability(qpol.QPOL_CAP_MLS): if self.policy.capability(qpol.QPOL_CAP_MLS):

View File

@ -1,4 +1,4 @@
# Copyright 2014, Tresys Technology, LLC # Copyright 2014-2015, Tresys Technology, LLC
# #
# This file is part of SETools. # This file is part of SETools.
# #
@ -21,6 +21,35 @@ import itertools
from . import qpol from . import qpol
from . import symbol from . import symbol
# qpol does not expose an equivalent of a sensitivity declaration.
# qpol_level_t is equivalent to the level declaration:
# level s0:c0.c1023;
# qpol_mls_level_t represents a level as used in contexts,
# such as range_transitions or labeling statements such as
# portcon and nodecon.
# Here qpol_level_t is also used for MLSSensitivity
# since it has the sensitivity name, dominance, and there
# is a 1:1 correspondence between the sensitivity declarations
# and level declarations.
class InvalidSensitivity(symbol.InvalidSymbol):
"""
Exception for an invalid sensitivity.
"""
pass
class InvalidLevel(symbol.InvalidSymbol):
"""
Exception for an invalid level.
"""
pass
class MLSDisabled(Exception): class MLSDisabled(Exception):
@ -41,17 +70,41 @@ def category_factory(policy, symbol):
def sensitivity_factory(policy, symbol): def sensitivity_factory(policy, symbol):
"""Factory function for creating MLS sensitivity objects.""" """Factory function for creating MLS sensitivity objects."""
raise NotImplementedError if isinstance(symbol, qpol.qpol_level_t):
return MLSSensitivity(policy, symbol)
try:
return MLSSensitivity(policy, qpol.qpol_level_t(policy, symbol))
except ValueError:
raise InvalidSensitivity("{0} is not a valid sensitivity".format(symbol))
def level_factory(policy, symbol): def level_factory(policy, symbol):
"""Factory function for creating MLS level objects.""" """
Factory function for creating MLS level objects (e.g. levels used
in contexts of labeling statements)
"""
if not isinstance(symbol, qpol.qpol_mls_level_t): if not isinstance(symbol, qpol.qpol_mls_level_t):
raise TypeError("MLS levels cannot be looked-up.") raise TypeError("MLS levels cannot be looked-up.")
return MLSLevel(policy, symbol) return MLSLevel(policy, symbol)
def level_decl_factory(policy, symbol):
"""
Factory function for creating MLS level declaration objects.
(level statements) Lookups are only by sensitivity name.
"""
if isinstance(symbol, qpol.qpol_level_t):
return MLSLevelDecl(policy, symbol)
try:
return MLSLevelDecl(policy, qpol.qpol_level_t(policy, symbol))
except ValueError:
raise InvalidLevel("{0} is not a valid sensitivity".format(symbol))
def range_factory(policy, symbol): def range_factory(policy, symbol):
"""Factory function for creating MLS range objects.""" """Factory function for creating MLS range objects."""
if not isinstance(symbol, qpol.qpol_mls_range_t): if not isinstance(symbol, qpol.qpol_mls_range_t):
@ -65,12 +118,7 @@ class MLSCategory(symbol.PolicySymbol):
"""An MLS category.""" """An MLS category."""
@property @property
def isalias(self): def _value(self):
"""(T/F) this is an alias."""
return self.qpol_symbol.isalias(self.policy)
@property
def value(self):
""" """
The value of the category. The value of the category.
@ -78,7 +126,7 @@ class MLSCategory(symbol.PolicySymbol):
be sorted based on their policy declaration order instead of be sorted based on their policy declaration order instead of
by their name. This has no other use. by their name. This has no other use.
Example usage: sorted(self.categories(), key=lambda k: k.value) Example usage: sorted(self.categories(), key=lambda k: k._value)
""" """
return self.qpol_symbol.value(self.policy) return self.qpol_symbol.value(self.policy)
@ -88,41 +136,83 @@ class MLSCategory(symbol.PolicySymbol):
for alias in self.qpol_symbol.alias_iter(self.policy): for alias in self.qpol_symbol.alias_iter(self.policy):
yield alias yield alias
# libqpol does not expose sensitivities as an individual component def statement(self):
return "category {0};".format(self)
class MLSSensitivity(symbol.PolicySymbol): class MLSSensitivity(symbol.PolicySymbol):
pass
"""An MLS sensitivity"""
class MLSLevel(symbol.PolicySymbol):
"""An MLS level."""
def __eq__(self, other): def __eq__(self, other):
if self.policy == other.policy: return (self._value == other._value)
if (self.qpol_symbol.sens_name(self.policy) !=
other.qpol_symbol.sens_name(self.policy)):
return False
selfcats = set(str(c) for c in self.categories()) def __ge__(self, other):
othercats = set(str(c) for c in other.categories()) return (self._value >= other._value)
return (not selfcats ^ othercats)
else: def __gt__(self, other):
raise NotImplementedError return (self._value > other._value)
def __le__(self, other):
return (self._value <= other._value)
def __lt__(self, other):
return (self._value < other._value)
@property
def _value(self):
"""
The value of the sensitivity.
This is a low-level policy detail exposed so that sensitivities can
be compared based on their dominance. This has no other use.
"""
return self.qpol_symbol.value(self.policy)
def statement(self):
return "sensitivity {0};".format(self)
class BaseMLSLevel(symbol.PolicySymbol):
"""Abstract base class for MLS levels."""
def __eq__(self, other):
selfcats = set(str(c) for c in self.categories())
othercats = set(str(c) for c in other.categories())
return (self.sensitivity == other.sensitivity and selfcats == othercats)
def __ge__(self, other):
selfcats = set(str(c) for c in self.categories())
othercats = set(str(c) for c in other.categories())
return (self.sensitivity >= other.sensitivity and selfcats >= othercats)
def __gt__(self, other):
selfcats = set(str(c) for c in self.categories())
othercats = set(str(c) for c in other.categories())
return (self.sensitivity > other.sensitivity and selfcats > othercats)
def __le__(self, other):
selfcats = set(str(c) for c in self.categories())
othercats = set(str(c) for c in other.categories())
return (self.sensitivity <= other.sensitivity and selfcats <= othercats)
def __lt__(self, other):
selfcats = set(str(c) for c in self.categories())
othercats = set(str(c) for c in other.categories())
return (self.sensitivity < other.sensitivity and selfcats < othercats)
def __str__(self): def __str__(self):
lvl = str(self.qpol_symbol.sens_name(self.policy)) lvl = str(self.sensitivity)
# sort by policy declaration order # sort by policy declaration order
cats = sorted(self.categories(), key=lambda k: k.value) cats = sorted(self.categories(), key=lambda k: k._value)
if cats: if cats:
# generate short category notation # generate short category notation
shortlist = [] shortlist = []
for k, g in itertools.groupby(cats, key=lambda k, for k, g in itertools.groupby(cats, key=lambda k,
c=itertools.count(): k.value - next(c)): c=itertools.count(): k._value - next(c)):
group = list(g) group = list(g)
if len(group) > 1: if len(group) > 1:
shortlist.append("{0}.{1}".format(group[0], group[-1])) shortlist.append("{0}.{1}".format(group[0], group[-1]))
@ -133,6 +223,10 @@ class MLSLevel(symbol.PolicySymbol):
return lvl return lvl
@property
def sensitivity(self):
raise NotImplementedError
def categories(self): def categories(self):
""" """
Generator that yields all individual categories for this level. Generator that yields all individual categories for this level.
@ -144,6 +238,35 @@ class MLSLevel(symbol.PolicySymbol):
yield category_factory(self.policy, cat) yield category_factory(self.policy, cat)
class MLSLevelDecl(BaseMLSLevel):
"""
The declaration statement for MLS levels, e.g:
level s7:c0.c1023;
"""
@property
def sensitivity(self):
"""The sensitivity of the level."""
# since the qpol symbol for levels is also used for
# MLSSensitivity objects, use self's qpol symbol
return sensitivity_factory(self.policy, self.qpol_symbol)
def statement(self):
return "level {0};".format(self)
class MLSLevel(BaseMLSLevel):
"""An MLS level used in contexts."""
@property
def sensitivity(self):
"""The sensitivity of the level."""
return sensitivity_factory(self.policy, self.qpol_symbol.sens_name(self.policy))
class MLSRange(symbol.PolicySymbol): class MLSRange(symbol.PolicySymbol):
"""An MLS range""" """An MLS range"""

View File

@ -398,6 +398,7 @@ typedef enum qpol_capability
}; };
%newobject level_iter(); %newobject level_iter();
%pythoncode %{ @QpolGenerator(_qpol.qpol_level_from_void) %}
qpol_iterator_t *level_iter() { qpol_iterator_t *level_iter() {
BEGIN_EXCEPTION BEGIN_EXCEPTION
qpol_iterator_t *iter; qpol_iterator_t *iter;

View File

@ -22,9 +22,8 @@ from . import qpol
class InvalidSymbol(Exception): class InvalidSymbol(Exception):
""" """
Exception for invalid symbols. Typically this is the case when Exception for invalid symbols. Typically this is attempting to
one symbol optionally relates to another, such as object classes look up an object in the policy, but it does not exist.
optionally inheriting a common.
""" """
pass pass