78dd35dd8c
Adds the policy openstack-common module and implements policy checks for the v2 API. Note that this cut only addresses whole objects (i.e., a subnet or a network or a port), not specific fields within objects. (This means that attributes are not filtered out based on policies.) Implements blueprint authorization-support-for-quantum. Change-Id: I1b52b1791a1f14f0af6508a63a40a38e440f15fe
239 lines
7.4 KiB
Python
239 lines
7.4 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
# Copyright (c) 2011 OpenStack, LLC.
|
|
# All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
"""Common Policy Engine Implementation"""
|
|
|
|
import json
|
|
import logging
|
|
import urllib
|
|
import urllib2
|
|
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
|
_BRAIN = None
|
|
|
|
|
|
def set_brain(brain):
|
|
"""Set the brain used by enforce().
|
|
|
|
Defaults use Brain() if not set.
|
|
|
|
"""
|
|
global _BRAIN
|
|
_BRAIN = brain
|
|
|
|
|
|
def reset():
|
|
"""Clear the brain used by enforce()."""
|
|
global _BRAIN
|
|
_BRAIN = None
|
|
|
|
|
|
def enforce(match_list, target_dict, credentials_dict, exc=None,
|
|
*args, **kwargs):
|
|
"""Enforces authorization of some rules against credentials.
|
|
|
|
:param match_list: nested tuples of data to match against
|
|
|
|
The basic brain supports three types of match lists:
|
|
|
|
1) rules
|
|
|
|
looks like: ``('rule:compute:get_instance',)``
|
|
|
|
Retrieves the named rule from the rules dict and recursively
|
|
checks against the contents of the rule.
|
|
|
|
2) roles
|
|
|
|
looks like: ``('role:compute:admin',)``
|
|
|
|
Matches if the specified role is in credentials_dict['roles'].
|
|
|
|
3) generic
|
|
|
|
looks like: ``('tenant_id:%(tenant_id)s',)``
|
|
|
|
Substitutes values from the target dict into the match using
|
|
the % operator and matches them against the creds dict.
|
|
|
|
Combining rules:
|
|
|
|
The brain returns True if any of the outer tuple of rules
|
|
match and also True if all of the inner tuples match. You
|
|
can use this to perform simple boolean logic. For
|
|
example, the following rule would return True if the creds
|
|
contain the role 'admin' OR the if the tenant_id matches
|
|
the target dict AND the the creds contains the role
|
|
'compute_sysadmin':
|
|
|
|
::
|
|
|
|
{
|
|
"rule:combined": (
|
|
'role:admin',
|
|
('tenant_id:%(tenant_id)s', 'role:compute_sysadmin')
|
|
)
|
|
}
|
|
|
|
Note that rule and role are reserved words in the credentials match, so
|
|
you can't match against properties with those names. Custom brains may
|
|
also add new reserved words. For example, the HttpBrain adds http as a
|
|
reserved word.
|
|
|
|
:param target_dict: dict of object properties
|
|
|
|
Target dicts contain as much information as we can about the object being
|
|
operated on.
|
|
|
|
:param credentials_dict: dict of actor properties
|
|
|
|
Credentials dicts contain as much information as we can about the user
|
|
performing the action.
|
|
|
|
:param exc: exception to raise
|
|
|
|
Class of the exception to raise if the check fails. Any remaining
|
|
arguments passed to enforce() (both positional and keyword arguments)
|
|
will be passed to the exception class. If exc is not provided, returns
|
|
False.
|
|
|
|
:return: True if the policy allows the action
|
|
:return: False if the policy does not allow the action and exc is not set
|
|
"""
|
|
global _BRAIN
|
|
if not _BRAIN:
|
|
_BRAIN = Brain()
|
|
if not _BRAIN.check(match_list, target_dict, credentials_dict):
|
|
if exc:
|
|
raise exc(*args, **kwargs)
|
|
return False
|
|
return True
|
|
|
|
|
|
class Brain(object):
|
|
"""Implements policy checking."""
|
|
@classmethod
|
|
def load_json(cls, data, default_rule=None):
|
|
"""Init a brain using json instead of a rules dictionary."""
|
|
rules_dict = json.loads(data)
|
|
return cls(rules=rules_dict, default_rule=default_rule)
|
|
|
|
def __init__(self, rules=None, default_rule=None):
|
|
self.rules = rules or {}
|
|
self.default_rule = default_rule
|
|
|
|
def add_rule(self, key, match):
|
|
self.rules[key] = match
|
|
|
|
def _check(self, match, target_dict, cred_dict):
|
|
try:
|
|
match_kind, match_value = match.split(':', 1)
|
|
except Exception:
|
|
LOG.exception(_("Failed to understand rule %(match)r") % locals())
|
|
# If the rule is invalid, fail closed
|
|
return False
|
|
try:
|
|
f = getattr(self, '_check_%s' % match_kind)
|
|
except AttributeError:
|
|
if not self._check_generic(match, target_dict, cred_dict):
|
|
return False
|
|
else:
|
|
if not f(match_value, target_dict, cred_dict):
|
|
return False
|
|
return True
|
|
|
|
def check(self, match_list, target_dict, cred_dict):
|
|
"""Checks authorization of some rules against credentials.
|
|
|
|
Detailed description of the check with examples in policy.enforce().
|
|
|
|
:param match_list: nested tuples of data to match against
|
|
:param target_dict: dict of object properties
|
|
:param credentials_dict: dict of actor properties
|
|
|
|
:returns: True if the check passes
|
|
|
|
"""
|
|
if not match_list:
|
|
return True
|
|
for and_list in match_list:
|
|
if isinstance(and_list, basestring):
|
|
and_list = (and_list,)
|
|
if all([self._check(item, target_dict, cred_dict)
|
|
for item in and_list]):
|
|
return True
|
|
return False
|
|
|
|
def _check_rule(self, match, target_dict, cred_dict):
|
|
"""Recursively checks credentials based on the brains rules."""
|
|
try:
|
|
new_match_list = self.rules[match]
|
|
except KeyError:
|
|
if self.default_rule and match != self.default_rule:
|
|
new_match_list = ('rule:%s' % self.default_rule,)
|
|
else:
|
|
return False
|
|
|
|
return self.check(new_match_list, target_dict, cred_dict)
|
|
|
|
def _check_role(self, match, target_dict, cred_dict):
|
|
"""Check that there is a matching role in the cred dict."""
|
|
return match.lower() in [x.lower() for x in cred_dict['roles']]
|
|
|
|
def _check_generic(self, match, target_dict, cred_dict):
|
|
"""Check an individual match.
|
|
|
|
Matches look like:
|
|
|
|
tenant:%(tenant_id)s
|
|
role:compute:admin
|
|
|
|
"""
|
|
|
|
# TODO(termie): do dict inspection via dot syntax
|
|
match = match % target_dict
|
|
key, value = match.split(':', 1)
|
|
if key in cred_dict:
|
|
return value == cred_dict[key]
|
|
return False
|
|
|
|
|
|
class HttpBrain(Brain):
|
|
"""A brain that can check external urls for policy.
|
|
|
|
Posts json blobs for target and credentials.
|
|
|
|
"""
|
|
|
|
def _check_http(self, match, target_dict, cred_dict):
|
|
"""Check http: rules by calling to a remote server.
|
|
|
|
This example implementation simply verifies that the response is
|
|
exactly 'True'. A custom brain using response codes could easily
|
|
be implemented.
|
|
|
|
"""
|
|
url = match % target_dict
|
|
data = {'target': json.dumps(target_dict),
|
|
'credentials': json.dumps(cred_dict)}
|
|
post_data = urllib.urlencode(data)
|
|
f = urllib2.urlopen(url, post_data)
|
|
return f.read() == "True"
|