mirror of
https://github.com/SELinuxProject/setools
synced 2025-04-01 22:58:12 +00:00
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:
parent
cf601f08fd
commit
880582fe73
@ -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."""
|
||||||
|
|
||||||
|
@ -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):
|
||||||
|
@ -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"""
|
||||||
|
@ -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;
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user