"""passbook core policy engine""" from logging import getLogger from celery import group from django.http import HttpRequest from passbook.core.celery import CELERY_APP from passbook.core.models import Policy, User LOGGER = getLogger(__name__) def get_remote_ip(request: HttpRequest) -> str: """Return the remote's IP""" if not request: return '0.0.0.0' # nosec if request.META.get('HTTP_X_FORWARDED_FOR'): return request.META.get('HTTP_X_FORWARDED_FOR') return request.META.get('REMOTE_ADDR') @CELERY_APP.task() def _policy_engine_task(user_pk, policy_pk, **kwargs): """Task wrapper to run policy checking""" policy_obj = Policy.objects.filter(pk=policy_pk).select_subclasses().first() user_obj = User.objects.get(pk=user_pk) for key, value in kwargs.items(): setattr(user_obj, key, value) LOGGER.debug("Running policy `%s`#%s for user %s...", policy_obj.name, policy_obj.pk.hex, user_obj) policy_result = policy_obj.passes(user_obj) # Handle policy result correctly if result, message or just result message = None if isinstance(policy_result, (tuple, list)): policy_result, message = policy_result # Invert result if policy.negate is set if policy_obj.negate: policy_result = not policy_result LOGGER.debug("Policy %r#%s got %s", policy_obj.name, policy_obj.pk.hex, policy_result) return policy_obj.action, policy_result, message class PolicyEngine: """Orchestrate policy checking, launch tasks and return result""" policies = None _group = None _request = None _user = None def __init__(self, policies): self.policies = policies self._request = None self._user = None def for_user(self, user): """Check policies for user""" self._user = user return self def with_request(self, request): """Set request""" self._request = request return self def build(self): """Build task group""" signatures = [] kwargs = { '__password__': getattr(self._user, '__password__', None), 'remote_ip': get_remote_ip(self._request) } for policy in self.policies: signatures.append(_policy_engine_task.s(self._user.pk, policy.pk.hex, **kwargs)) self._group = group(signatures)() return self @property def result(self): """Get policy-checking result""" messages = [] for policy_action, policy_result, policy_message in self._group.get(): passing = (policy_action == Policy.ACTION_ALLOW and policy_result) or \ (policy_action == Policy.ACTION_DENY and not policy_result) if policy_message: messages.append(policy_message) if not passing: return False, messages return True, messages