Policies for external networks

Bug #1042030 , part 2

Also reworks model queries in order to allow plugins and extensions
to augment them as required through hooks.

Change-Id: Ice72fc6d3b1c613d596c037818ed66d7e9ed841d
This commit is contained in:
Salvatore Orlando 2012-09-03 14:17:20 -07:00
parent 4ce397d02c
commit a7326a947b
9 changed files with 310 additions and 141 deletions

View File

@ -3,6 +3,8 @@
"admin_or_network_owner": [["role:admin"], ["tenant_id:%(network_tenant_id)s"]], "admin_or_network_owner": [["role:admin"], ["tenant_id:%(network_tenant_id)s"]],
"admin_only": [["role:admin"]], "admin_only": [["role:admin"]],
"regular_user": [], "regular_user": [],
"shared": [["field:networks:shared=True"]],
"external": [["field:networks:router:external=True"]],
"default": [["rule:admin_or_owner"]], "default": [["rule:admin_or_owner"]],
"extension:provider_network:view": [["rule:admin_only"]], "extension:provider_network:view": [["rule:admin_only"]],
@ -11,27 +13,22 @@
"extension:router:view": [["rule:regular_user"]], "extension:router:view": [["rule:regular_user"]],
"extension:router:set": [["rule:admin_only"]], "extension:router:set": [["rule:admin_only"]],
"networks:private:read": [["rule:admin_or_owner"]],
"networks:private:write": [["rule:admin_or_owner"]],
"networks:shared:read": [["rule:regular_user"]],
"networks:shared:write": [["rule:admin_only"]],
"subnets:private:read": [["rule:admin_or_owner"]], "subnets:private:read": [["rule:admin_or_owner"]],
"subnets:private:write": [["rule:admin_or_owner"]], "subnets:private:write": [["rule:admin_or_owner"]],
"subnets:shared:read": [["rule:regular_user"]], "subnets:shared:read": [["rule:regular_user"]],
"subnets:shared:write": [["rule:admin_only"]], "subnets:shared:write": [["rule:admin_only"]],
"create_subnet": [["rule:admin_or_network_owner"]], "create_subnet": [["rule:admin_or_network_owner"]],
"get_subnet": [], "get_subnet": [["rule:admin_or_owner"], ["rule:shared"]],
"update_subnet": [["rule:admin_or_network_owner"]], "update_subnet": [["rule:admin_or_network_owner"]],
"delete_subnet": [["rule:admin_or_network_owner"]], "delete_subnet": [["rule:admin_or_network_owner"]],
"create_network": [], "create_network": [],
"get_network": [], "get_network": [["rule:admin_or_owner"], ["rule:shared"], ["rule:external"]],
"create_network:shared": [["rule:admin_only"]], "create_network:shared": [["rule:admin_only"]],
"update_network": [], "create_network:router:external": [["rule:admin_only"]],
"update_network:shared": [["rule:admin_only"]], "update_network": [["rule:admin_or_owner"]],
"delete_network": [], "delete_network": [["rule:admin_or_owner"]],
"create_port": [], "create_port": [],
"create_port:mac_address": [["rule:admin_or_network_owner"]], "create_port:mac_address": [["rule:admin_or_network_owner"]],

View File

@ -43,6 +43,8 @@ FAULT_MAP = {exceptions.NotFound: webob.exc.HTTPNotFound,
exceptions.InvalidSharedSetting: webob.exc.HTTPConflict, exceptions.InvalidSharedSetting: webob.exc.HTTPConflict,
exceptions.HostRoutesExhausted: webob.exc.HTTPBadRequest, exceptions.HostRoutesExhausted: webob.exc.HTTPBadRequest,
exceptions.DNSNameServersExhausted: webob.exc.HTTPBadRequest, exceptions.DNSNameServersExhausted: webob.exc.HTTPBadRequest,
# Some plugins enforce policies as well
exceptions.PolicyNotAuthorized: webob.exc.HTTPForbidden
} }
QUOTAS = quota.QUOTAS QUOTAS = quota.QUOTAS

View File

@ -50,6 +50,12 @@ class QuantumDbPluginV2(quantum_plugin_base_v2.QuantumPluginBaseV2):
# bulk operations. Name mangling is used in order to ensure it # bulk operations. Name mangling is used in order to ensure it
# is qualified by class # is qualified by class
__native_bulk_support = True __native_bulk_support = True
# Plugins, mixin classes implementing extension will register
# hooks into the dict below for "augmenting" the "core way" of
# building a query for retrieving objects from a model class.
# To this aim, the register_model_query_hook and unregister_query_hook
# from this class should be invoked
_model_query_hooks = {}
def __init__(self): def __init__(self):
# NOTE(jkoelker) This is an incomlete implementation. Subclasses # NOTE(jkoelker) This is an incomlete implementation. Subclasses
@ -73,20 +79,58 @@ class QuantumDbPluginV2(quantum_plugin_base_v2.QuantumPluginBaseV2):
def _model_query(self, context, model): def _model_query(self, context, model):
query = context.session.query(model) query = context.session.query(model)
# define basic filter condition for model query
# NOTE(jkoelker) non-admin queries are scoped to their tenant_id # NOTE(jkoelker) non-admin queries are scoped to their tenant_id
# NOTE(salvatore-orlando): unless the model allows for shared objects # NOTE(salvatore-orlando): unless the model allows for shared objects
query_filter = None
if not context.is_admin and hasattr(model, 'tenant_id'): if not context.is_admin and hasattr(model, 'tenant_id'):
if hasattr(model, 'shared'): if hasattr(model, 'shared'):
query = query.filter((model.tenant_id == context.tenant_id) | query_filter = ((model.tenant_id == context.tenant_id) |
(model.shared)) (model.shared))
else: else:
query = query.filter(model.tenant_id == context.tenant_id) query_filter = (model.tenant_id == context.tenant_id)
# Execute query hooks registered from mixins and plugins
for _name, hooks in self._model_query_hooks.get(model,
{}).iteritems():
query_hook = hooks.get('query')
filter_hook = hooks.get('filter')
if query_hook:
query = query_hook(self, context, model, query)
if filter_hook:
query_filter = filter_hook(self, context, model, query_filter)
# NOTE(salvatore-orlando): 'if query_filter' will try to evaluate the
# condition, raising an exception
if query_filter is not None:
query = query.filter(query_filter)
return query return query
@classmethod
def register_model_query_hook(cls, model, name, query_hook, filter_hook):
""" register an hook to be invoked when a query is executed.
Add the hooks to the _model_query_hooks dict. Models are the keys
of this dict, whereas the value is another dict mapping hook names to
callables performing the hook.
Each hook has a "query" component, used to build the query expression
and a "filter" component, which is used to build the filter expression.
Query hooks take as input the query being built and return a
transformed query expression.
Filter hooks take as input the filter expression being built and return
a transformed filter expression
"""
model_hooks = cls._model_query_hooks.get(model)
if not model_hooks:
# add key to dict
model_hooks = {}
cls._model_query_hooks[model] = model_hooks
model_hooks[name] = {'query': query_hook, 'filter': filter_hook}
def _get_by_id(self, context, model, id): def _get_by_id(self, context, model, id):
query = self._model_query(context, model) query = self._model_query(context, model)
return query.filter_by(id=id).one() return query.filter(model.id == id).one()
def _get_network(self, context, id): def _get_network(self, context, id):
try: try:

View File

@ -25,11 +25,13 @@ import netaddr
import sqlalchemy as sa import sqlalchemy as sa
from sqlalchemy import orm from sqlalchemy import orm
from sqlalchemy.orm import exc from sqlalchemy.orm import exc
from sqlalchemy.sql import expression as expr
import webob.exc as w_exc import webob.exc as w_exc
from quantum.api.v2 import attributes from quantum.api.v2 import attributes
from quantum.common import exceptions as q_exc from quantum.common import exceptions as q_exc
from quantum.common import utils from quantum.common import utils
from quantum.db import db_base_plugin_v2
from quantum.db import model_base from quantum.db import model_base
from quantum.db import models_v2 from quantum.db import models_v2
from quantum.extensions import l3 from quantum.extensions import l3
@ -57,8 +59,7 @@ class Router(model_base.BASEV2, models_v2.HasId, models_v2.HasTenant):
name = sa.Column(sa.String(255)) name = sa.Column(sa.String(255))
status = sa.Column(sa.String(16)) status = sa.Column(sa.String(16))
admin_state_up = sa.Column(sa.Boolean) admin_state_up = sa.Column(sa.Boolean)
gw_port_id = sa.Column(sa.String(36), sa.ForeignKey('ports.id', gw_port_id = sa.Column(sa.String(36), sa.ForeignKey('ports.id'))
ondelete="CASCADE"))
gw_port = orm.relationship(models_v2.Port) gw_port = orm.relationship(models_v2.Port)
@ -85,6 +86,30 @@ class FloatingIP(model_base.BASEV2, models_v2.HasId, models_v2.HasTenant):
class L3_NAT_db_mixin(l3.RouterPluginBase): class L3_NAT_db_mixin(l3.RouterPluginBase):
"""Mixin class to add L3/NAT router methods to db_plugin_base_v2""" """Mixin class to add L3/NAT router methods to db_plugin_base_v2"""
def _network_model_hook(self, context, original_model, query):
query = query.outerjoin(ExternalNetwork,
(original_model.id ==
ExternalNetwork.network_id))
return query
def _network_filter_hook(self, context, original_model, conditions):
if conditions is not None and not hasattr(conditions, '__iter__'):
conditions = (conditions, )
# Apply the external network filter only in non-admin context
if not context.is_admin and hasattr(original_model, 'tenant_id'):
conditions = expr.or_(ExternalNetwork.network_id != expr.null(),
*conditions)
return conditions
# TODO(salvatore-orlando): Perform this operation without explicitly
# referring to db_base_plugin_v2, as plugins that do not extend from it
# might exist in the future
db_base_plugin_v2.QuantumDbPluginV2.register_model_query_hook(
models_v2.Network,
"external_net",
_network_model_hook,
_network_filter_hook)
def _get_router(self, context, id): def _get_router(self, context, id):
try: try:
router = self._get_by_id(context, Router, id) router = self._get_by_id(context, Router, id)
@ -163,14 +188,16 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
with context.session.begin(subtransactions=True): with context.session.begin(subtransactions=True):
router.update({'gw_port_id': None}) router.update({'gw_port_id': None})
context.session.add(router) context.session.add(router)
self.delete_port(context, gw_port['id'], l3_port_check=False) self.delete_port(context.elevated(), gw_port['id'],
l3_port_check=False)
if network_id is not None and (gw_port is None or if network_id is not None and (gw_port is None or
gw_port['network_id'] != network_id): gw_port['network_id'] != network_id):
# Port has no 'tenant-id', as it is hidden from user # Port has no 'tenant-id', as it is hidden from user
gw_port = self.create_port(context, { gw_port = self.create_port(context.elevated(), {
'port': 'port':
{'network_id': network_id, {'tenant_id': '', # intentionally not set
'network_id': network_id,
'mac_address': attributes.ATTR_NOT_SPECIFIED, 'mac_address': attributes.ATTR_NOT_SPECIFIED,
'fixed_ips': attributes.ATTR_NOT_SPECIFIED, 'fixed_ips': attributes.ATTR_NOT_SPECIFIED,
'device_id': router_id, 'device_id': router_id,
@ -179,7 +206,8 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
'name': ''}}) 'name': ''}})
if not len(gw_port['fixed_ips']): if not len(gw_port['fixed_ips']):
self.delete_port(context, gw_port['id'], l3_port_check=False) self.delete_port(context.elevated(), gw_port['id'],
l3_port_check=False)
msg = ('No IPs available for external network %s' % msg = ('No IPs available for external network %s' %
network_id) network_id)
raise q_exc.BadRequest(resource='router', msg=msg) raise q_exc.BadRequest(resource='router', msg=msg)
@ -197,8 +225,14 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
ports = self.get_ports(context, filters=device_filter) ports = self.get_ports(context, filters=device_filter)
if ports: if ports:
raise l3.RouterInUse(router_id=id) raise l3.RouterInUse(router_id=id)
# NOTE(salvatore-orlando): gw port will be automatically deleted
# thanks to cascading on the ORM relationship # delete any gw port
device_filter = {'device_id': [id],
'device_owner': [DEVICE_OWNER_ROUTER_GW]}
ports = self.get_ports(context.elevated(), filters=device_filter)
if ports:
self._delete_port(context.elevated(), ports[0]['id'])
context.session.delete(router) context.session.delete(router)
def get_router(self, context, id, fields=None): def get_router(self, context, id, fields=None):
@ -279,7 +313,8 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
'subnet_id': subnet['id']} 'subnet_id': subnet['id']}
port = self.create_port(context, { port = self.create_port(context, {
'port': 'port':
{'network_id': subnet['network_id'], {'tenant_id': subnet['tenant_id'],
'network_id': subnet['network_id'],
'fixed_ips': [fixed_ip], 'fixed_ips': [fixed_ip],
'mac_address': attributes.ATTR_NOT_SPECIFIED, 'mac_address': attributes.ATTR_NOT_SPECIFIED,
'admin_state_up': True, 'admin_state_up': True,
@ -434,7 +469,7 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
# confirm that this router has a floating # confirm that this router has a floating
# ip enabled gateway with support for this floating IP network # ip enabled gateway with support for this floating IP network
try: try:
port_qry = context.session.query(models_v2.Port) port_qry = context.elevated().session.query(models_v2.Port)
ports = port_qry.filter_by( ports = port_qry.filter_by(
network_id=floating_network_id, network_id=floating_network_id,
device_id=router_id, device_id=router_id,
@ -448,6 +483,10 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
def _update_fip_assoc(self, context, fip, floatingip_db, external_port): def _update_fip_assoc(self, context, fip, floatingip_db, external_port):
port_id = internal_ip_address = router_id = None port_id = internal_ip_address = router_id = None
if (('fixed_ip_address' in fip and fip['fixed_ip_address']) and
not ('port_id' in fip and fip['port_id'])):
msg = "fixed_ip_address cannot be specified without a port_id"
raise q_exc.BadRequest(resource='floatingip', msg=msg)
if 'port_id' in fip and fip['port_id']: if 'port_id' in fip and fip['port_id']:
port_qry = context.session.query(FloatingIP) port_qry = context.session.query(FloatingIP)
try: try:
@ -477,9 +516,10 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
# This external port is never exposed to the tenant. # This external port is never exposed to the tenant.
# it is used purely for internal system and admin use when # it is used purely for internal system and admin use when
# managing floating IPs. # managing floating IPs.
external_port = self.create_port(context, { external_port = self.create_port(context.elevated(), {
'port': 'port':
{'network_id': f_net_id, {'tenant_id': '', # tenant intentionally not set
'network_id': f_net_id,
'mac_address': attributes.ATTR_NOT_SPECIFIED, 'mac_address': attributes.ATTR_NOT_SPECIFIED,
'fixed_ips': attributes.ATTR_NOT_SPECIFIED, 'fixed_ips': attributes.ATTR_NOT_SPECIFIED,
'admin_state_up': True, 'admin_state_up': True,
@ -490,7 +530,8 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
if not external_port['fixed_ips']: if not external_port['fixed_ips']:
msg = "Unable to find any IP address on external network" msg = "Unable to find any IP address on external network"
# remove the external port # remove the external port
self.delete_port(context, external_port['id'], l3_port_check=False) self.delete_port(context.elevated(), external_port['id'],
l3_port_check=False)
raise q_exc.BadRequest(resource='floatingip', msg=msg) raise q_exc.BadRequest(resource='floatingip', msg=msg)
floating_ip_address = external_port['fixed_ips'][0]['ip_address'] floating_ip_address = external_port['fixed_ips'][0]['ip_address']
@ -510,6 +551,10 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
context.session.add(floatingip_db) context.session.add(floatingip_db)
# TODO(salvatore-orlando): Avoid broad catch # TODO(salvatore-orlando): Avoid broad catch
# Maybe by introducing base class for L3 exceptions # Maybe by introducing base class for L3 exceptions
except q_exc.BadRequest:
LOG.exception("Unable to create Floating ip due to a "
"malformed request")
raise
except Exception: except Exception:
LOG.exception("Floating IP association failed") LOG.exception("Floating IP association failed")
# Remove the port created for internal purposes # Remove the port created for internal purposes
@ -526,14 +571,16 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
fip['id'] = id fip['id'] = id
fip_port_id = floatingip_db['floating_port_id'] fip_port_id = floatingip_db['floating_port_id']
self._update_fip_assoc(context, fip, floatingip_db, self._update_fip_assoc(context, fip, floatingip_db,
self.get_port(context, fip_port_id)) self.get_port(context.elevated(),
fip_port_id))
return self._make_floatingip_dict(floatingip_db) return self._make_floatingip_dict(floatingip_db)
def delete_floatingip(self, context, id): def delete_floatingip(self, context, id):
floatingip = self._get_floatingip(context, id) floatingip = self._get_floatingip(context, id)
with context.session.begin(subtransactions=True): with context.session.begin(subtransactions=True):
context.session.delete(floatingip) context.session.delete(floatingip)
self.delete_port(context, floatingip['floating_port_id'], self.delete_port(context.elevated(),
floatingip['floating_port_id'],
l3_port_check=False) l3_port_check=False)
def get_floatingip(self, context, id, fields=None): def get_floatingip(self, context, id, fields=None):
@ -595,11 +642,11 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
def _extend_network_dict_l3(self, context, network): def _extend_network_dict_l3(self, context, network):
if self._check_l3_view_auth(context, network): if self._check_l3_view_auth(context, network):
network['router:external'] = self._network_is_external( network[l3.EXTERNAL] = self._network_is_external(
context, network['id']) context, network['id'])
def _process_l3_create(self, context, net_data, net_id): def _process_l3_create(self, context, net_data, net_id):
external = net_data.get('router:external') external = net_data.get(l3.EXTERNAL)
external_set = attributes.is_attr_set(external) external_set = attributes.is_attr_set(external)
if not external_set: if not external_set:
@ -613,7 +660,7 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
def _process_l3_update(self, context, net_data, net_id): def _process_l3_update(self, context, net_data, net_id):
new_value = net_data.get('router:external') new_value = net_data.get(l3.EXTERNAL)
if not attributes.is_attr_set(new_value): if not attributes.is_attr_set(new_value):
return return
@ -633,7 +680,7 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
context.session.query(models_v2.Port).filter_by( context.session.query(models_v2.Port).filter_by(
device_owner=DEVICE_OWNER_ROUTER_GW, device_owner=DEVICE_OWNER_ROUTER_GW,
network_id=net_id).first() network_id=net_id).first()
raise ExternalNetworkInUse(net_id=net_id) raise l3.ExternalNetworkInUse(net_id=net_id)
except exc.NoResultFound: except exc.NoResultFound:
pass # expected pass # expected

View File

@ -114,13 +114,16 @@ RESOURCE_ATTRIBUTE_MAP = {
}, },
} }
EXTERNAL = 'router:external'
EXTENDED_ATTRIBUTES_2_0 = { EXTENDED_ATTRIBUTES_2_0 = {
'networks': {'router:external': {'allow_post': True, 'networks': {EXTERNAL: {'allow_post': True,
'allow_put': True, 'allow_put': True,
'default': attr.ATTR_NOT_SPECIFIED, 'default': attr.ATTR_NOT_SPECIFIED,
'is_visible': True, 'is_visible': True,
'convert_to': attr.convert_to_boolean, 'convert_to': attr.convert_to_boolean,
'validate': {'type:boolean': None}}}} 'validate': {'type:boolean': None},
'enforce_policy': True,
'required_by_policy': True}}}
l3_quota_opts = [ l3_quota_opts = [
cfg.IntOpt('quota_router', cfg.IntOpt('quota_router',

View File

@ -18,12 +18,16 @@
""" """
Policy engine for quantum. Largely copied from nova. Policy engine for quantum. Largely copied from nova.
""" """
import logging
from quantum.api.v2 import attributes from quantum.api.v2 import attributes
from quantum.common import exceptions from quantum.common import exceptions
from quantum.openstack.common import cfg from quantum.openstack.common import cfg
import quantum.common.utils as utils import quantum.common.utils as utils
from quantum.openstack.common import policy from quantum.openstack.common import policy
LOG = logging.getLogger(__name__)
_POLICY_PATH = None _POLICY_PATH = None
_POLICY_CACHE = {} _POLICY_CACHE = {}
@ -49,16 +53,17 @@ def init():
reload_func=_set_brain) reload_func=_set_brain)
def get_resource_and_action(action):
""" Extract resource and action (write, read) from api operation """
data = action.split(':', 1)[0].split('_', 1)
return ("%ss" % data[-1], data[0] != 'get')
def _set_brain(data): def _set_brain(data):
default_rule = 'default' default_rule = 'default'
policy.set_brain(policy.Brain.load_json(data, default_rule)) policy.set_brain(policy.Brain.load_json(data, default_rule))
def _get_resource_and_action(action):
data = action.split(':', 1)[0].split('_', 1)
return ("%ss" % data[-1], data[0] != 'get')
def _is_attribute_explicitly_set(attribute_name, resource, target): def _is_attribute_explicitly_set(attribute_name, resource, target):
"""Verify that an attribute is present and has a non-default value""" """Verify that an attribute is present and has a non-default value"""
if ('default' in resource[attribute_name] and if ('default' in resource[attribute_name] and
@ -76,7 +81,7 @@ def _build_target(action, original_target, plugin, context):
"parent" resource of the targeted one. "parent" resource of the targeted one.
""" """
target = original_target.copy() target = original_target.copy()
resource, _w = _get_resource_and_action(action) resource, _a = get_resource_and_action(action)
hierarchy_info = attributes.RESOURCE_HIERARCHY_MAP.get(resource, None) hierarchy_info = attributes.RESOURCE_HIERARCHY_MAP.get(resource, None)
if hierarchy_info and plugin: if hierarchy_info and plugin:
# use the 'singular' version of the resource name # use the 'singular' version of the resource name
@ -90,31 +95,6 @@ def _build_target(action, original_target, plugin, context):
return target return target
def _create_access_rule_match(resource, is_write, shared):
if shared == resource[attributes.SHARED]:
return ('rule:%s:%s:%s' % (resource,
shared and 'shared' or 'private',
is_write and 'write' or 'read'), )
def _build_perm_match(action, target):
"""Create the permission rule match.
Given the current access right on a network (shared/private), and
the type of the current operation (read/write), builds a match
rule of the type <resource>:<sharing_mode>:<operation_type>
"""
resource, is_write = _get_resource_and_action(action)
res_map = attributes.RESOURCE_ATTRIBUTE_MAP
if (resource in res_map and
attributes.SHARED in res_map[resource] and
attributes.SHARED in target):
return ('rule:%s:%s:%s' % (resource,
target[attributes.SHARED]
and 'shared' or 'private',
is_write and 'write' or 'read'), )
def _build_match_list(action, target): def _build_match_list(action, target):
"""Create the list of rules to match for a given action. """Create the list of rules to match for a given action.
@ -127,7 +107,8 @@ def _build_match_list(action, target):
""" """
match_list = ('rule:%s' % action,) match_list = ('rule:%s' % action,)
resource, is_write = _get_resource_and_action(action) resource, is_write = get_resource_and_action(action)
if is_write:
# assigning to variable with short name for improving readability # assigning to variable with short name for improving readability
res_map = attributes.RESOURCE_ATTRIBUTE_MAP res_map = attributes.RESOURCE_ATTRIBUTE_MAP
if resource in res_map: if resource in res_map:
@ -139,14 +120,37 @@ def _build_match_list(action, target):
if 'enforce_policy' in attribute and is_write: if 'enforce_policy' in attribute and is_write:
match_list += ('rule:%s:%s' % (action, match_list += ('rule:%s:%s' % (action,
attribute_name),) attribute_name),)
# add permission-based rule (for shared resources)
perm_match = _build_perm_match(action, target)
if perm_match:
match_list += perm_match
# the policy engine must AND between all the rules
return [match_list] return [match_list]
@policy.register('field')
def check_field(brain, match_kind, match, target_dict, cred_dict):
# If this method is invoked for the wrong kind of match
# which should never happen, just skip the check and don't
# fail the policy evaluation
if match_kind != 'field':
LOG.warning("Field check function invoked with wrong match_kind:%s",
match_kind)
return True
resource, field_value = match.split(':', 1)
field, value = field_value.split('=', 1)
target_value = target_dict.get(field)
# target_value might be a boolean, explicitly compare with None
if target_value is None:
LOG.debug("Unable to find requested field: %s in target: %s",
field, target_dict)
return False
# Value migth need conversion - we need help from the attribute map
conv_func = attributes.RESOURCE_ATTRIBUTE_MAP[resource][field].get(
'convert_to', lambda x: x)
if target_value != conv_func(value):
LOG.debug("%s does not match the value in the target object:%s",
conv_func(value), target_value)
return False
# If we manage to get here, the policy check is successful
return True
def check(context, action, target, plugin=None): def check(context, action, target, plugin=None):
"""Verifies that the action is valid on the target in this context. """Verifies that the action is valid on the target in this context.
@ -184,10 +188,8 @@ def enforce(context, action, target, plugin=None):
""" """
init() init()
real_target = _build_target(action, target, plugin, context) real_target = _build_target(action, target, plugin, context)
match_list = _build_match_list(action, real_target) match_list = _build_match_list(action, real_target)
credentials = context.to_dict() credentials = context.to_dict()
policy.enforce(match_list, real_target, credentials, policy.enforce(match_list, real_target, credentials,
exceptions.PolicyNotAuthorized, action=action) exceptions.PolicyNotAuthorized, action=action)

View File

@ -184,11 +184,13 @@ class QuantumDbPluginV2TestCase(unittest2.TestCase):
req.environ['quantum.context'] = kwargs['context'] req.environ['quantum.context'] = kwargs['context']
return req.get_response(self.api) return req.get_response(self.api)
def _create_network(self, fmt, name, admin_status_up, **kwargs): def _create_network(self, fmt, name, admin_status_up,
arg_list=None, **kwargs):
data = {'network': {'name': name, data = {'network': {'name': name,
'admin_state_up': admin_status_up, 'admin_state_up': admin_status_up,
'tenant_id': self._tenant_id}} 'tenant_id': self._tenant_id}}
for arg in ('admin_state_up', 'tenant_id', 'shared'): for arg in (('admin_state_up', 'tenant_id', 'shared') +
(arg_list or ())):
# Arg must be present and not empty # Arg must be present and not empty
if arg in kwargs and kwargs[arg]: if arg in kwargs and kwargs[arg]:
data['network'][arg] = kwargs[arg] data['network'][arg] = kwargs[arg]
@ -315,10 +317,6 @@ class QuantumDbPluginV2TestCase(unittest2.TestCase):
raise webob.exc.HTTPClientError(code=res.status_int) raise webob.exc.HTTPClientError(code=res.status_int)
return self.deserialize(fmt, res) return self.deserialize(fmt, res)
def _make_port(self, fmt, net_id, **kwargs):
res = self._create_port(fmt, net_id, **kwargs)
return self.deserialize(fmt, res)
def _api_for_resource(self, resource): def _api_for_resource(self, resource):
if resource in ['networks', 'subnets', 'ports']: if resource in ['networks', 'subnets', 'ports']:
return self.api return self.api
@ -440,15 +438,24 @@ class QuantumDbPluginV2TestCase(unittest2.TestCase):
if not subnet: if not subnet:
with self.subnet() as subnet: with self.subnet() as subnet:
net_id = subnet['subnet']['network_id'] net_id = subnet['subnet']['network_id']
port = self._make_port(fmt, net_id, fixed_ips=fixed_ips, res = self._create_port(fmt, net_id, **kwargs)
**kwargs) port = self.deserialize(fmt, res)
# Things can go wrong - raise HTTP exc with res code only
# so it can be caught by unit tests
if res.status_int >= 400:
raise webob.exc.HTTPClientError(code=res.status_int)
yield port yield port
if not no_delete: if not no_delete:
self._delete('ports', port['port']['id']) self._delete('ports', port['port']['id'])
else: else:
net_id = subnet['subnet']['network_id'] net_id = subnet['subnet']['network_id']
port = self._make_port(fmt, net_id, fixed_ips=fixed_ips, res = self._create_port(fmt, net_id, **kwargs)
**kwargs) port = self.deserialize(fmt, res)
# Things can go wrong - raise HTTP exc with res code only
# so it can be caught by unit tests
if res.status_int >= 400:
raise webob.exc.HTTPClientError(code=res.status_int)
yield port yield port
if not no_delete: if not no_delete:
self._delete('ports', port['port']['id']) self._delete('ports', port['port']['id'])
@ -801,7 +808,7 @@ class TestPortsV2(QuantumDbPluginV2TestCase):
admin_status_up=True) admin_status_up=True)
network = self.deserialize(fmt, res) network = self.deserialize(fmt, res)
network_id = network['network']['id'] network_id = network['network']['id']
port = self._make_port(fmt, network_id, device_owner='network:dhcp') self._create_port(fmt, network_id, device_owner='network:dhcp')
req = self.new_delete_request('networks', network_id) req = self.new_delete_request('networks', network_id)
res = req.get_response(self.api) res = req.get_response(self.api)
self.assertEquals(res.status_int, 204) self.assertEquals(res.status_int, 204)
@ -1822,7 +1829,8 @@ class TestSubnetsV2(QuantumDbPluginV2TestCase):
network_id = network['network']['id'] network_id = network['network']['id']
subnet = self._make_subnet(fmt, network, gateway_ip, subnet = self._make_subnet(fmt, network, gateway_ip,
cidr, ip_version=4) cidr, ip_version=4)
port = self._make_port(fmt, network['network']['id'], self._create_port(fmt,
network['network']['id'],
device_owner='network:dhcp') device_owner='network:dhcp')
req = self.new_delete_request('subnets', subnet['subnet']['id']) req = self.new_delete_request('subnets', subnet['subnet']['id'])
res = req.get_response(self.api) res = req.get_response(self.api)

View File

@ -21,6 +21,7 @@
import contextlib import contextlib
import copy import copy
import itertools
import logging import logging
import unittest import unittest
@ -32,8 +33,10 @@ from quantum.api.v2 import attributes
from quantum.common import config from quantum.common import config
from quantum.common.test_lib import test_config from quantum.common.test_lib import test_config
from quantum.common import utils from quantum.common import utils
from quantum import context
from quantum.db import db_base_plugin_v2 from quantum.db import db_base_plugin_v2
from quantum.db import l3_db from quantum.db import l3_db
from quantum.db import models_v2
from quantum.extensions import extensions from quantum.extensions import extensions
from quantum.extensions import l3 from quantum.extensions import l3
from quantum import manager from quantum import manager
@ -267,6 +270,20 @@ class TestL3NatPlugin(db_base_plugin_v2.QuantumDbPluginV2,
class L3NatDBTestCase(test_db_plugin.QuantumDbPluginV2TestCase): class L3NatDBTestCase(test_db_plugin.QuantumDbPluginV2TestCase):
def _create_network(self, fmt, name, admin_status_up, **kwargs):
""" Override the routine for allowing the router:external attribute """
# attributes containing a colon should be passed with
# a double underscore
new_args = dict(itertools.izip(map(lambda x: x.replace('__', ':'),
kwargs),
kwargs.values()))
arg_list = (l3.EXTERNAL,)
return super(L3NatDBTestCase, self)._create_network(fmt,
name,
admin_status_up,
arg_list=arg_list,
**new_args)
def setUp(self): def setUp(self):
test_config['plugin_name_v2'] = ( test_config['plugin_name_v2'] = (
'quantum.tests.unit.test_l3_plugin.TestL3NatPlugin') 'quantum.tests.unit.test_l3_plugin.TestL3NatPlugin')
@ -567,7 +584,7 @@ class L3NatDBTestCase(test_db_plugin.QuantumDbPluginV2TestCase):
def _set_net_external(self, net_id): def _set_net_external(self, net_id):
self._update('networks', net_id, self._update('networks', net_id,
{'network': {'router:external': True}}) {'network': {l3.EXTERNAL: True}})
def _create_floatingip(self, fmt, network_id, port_id=None, def _create_floatingip(self, fmt, network_id, port_id=None,
fixed_ip=None): fixed_ip=None):
@ -794,9 +811,57 @@ class L3NatDBTestCase(test_db_plugin.QuantumDbPluginV2TestCase):
self.assertEquals(len(body['networks']), 2) self.assertEquals(len(body['networks']), 2)
body = self._list('networks', body = self._list('networks',
query_params="router:external=True") query_params="%s=True" % l3.EXTERNAL)
self.assertEquals(len(body['networks']), 1) self.assertEquals(len(body['networks']), 1)
body = self._list('networks', body = self._list('networks',
query_params="router:external=False") query_params="%s=False" % l3.EXTERNAL)
self.assertEquals(len(body['networks']), 1) self.assertEquals(len(body['networks']), 1)
def test_network_filter_hook_admin_context(self):
plugin = manager.QuantumManager.get_plugin()
ctx = context.Context(None, None, is_admin=True)
model = models_v2.Network
conditions = plugin._network_filter_hook(ctx, model, [])
self.assertEqual(conditions, [])
def test_network_filter_hook_nonadmin_context(self):
plugin = manager.QuantumManager.get_plugin()
ctx = context.Context('edinson', 'cavani')
model = models_v2.Network
txt = "externalnetworks.network_id IS NOT NULL"
conditions = plugin._network_filter_hook(ctx, model, [])
self.assertEqual(conditions.__str__(), txt)
# Try to concatenate confitions
conditions = plugin._network_filter_hook(ctx, model, conditions)
self.assertEqual(conditions.__str__(), "%s OR %s" % (txt, txt))
def test_create_port_external_network_non_admin_fails(self):
with self.network(router__external=True) as ext_net:
with self.subnet(network=ext_net) as ext_subnet:
with self.assertRaises(exc.HTTPClientError) as ctx_manager:
with self.port(subnet=ext_subnet,
set_context='True',
tenant_id='noadmin'):
pass
self.assertEquals(ctx_manager.exception.code, 403)
def test_create_port_external_network_admin_suceeds(self):
with self.network(router__external=True) as ext_net:
with self.subnet(network=ext_net) as ext_subnet:
with self.port(subnet=ext_subnet) as port:
self.assertEqual(port['port']['network_id'],
ext_net['network']['id'])
def test_create_external_network_non_admin_fails(self):
with self.assertRaises(exc.HTTPClientError) as ctx_manager:
with self.network(router__external=True,
set_context='True',
tenant_id='noadmin'):
pass
self.assertEquals(ctx_manager.exception.code, 403)
def test_create_external_network_admin_suceeds(self):
with self.network(router__external=True) as ext_net:
self.assertEqual(ext_net['network'][l3.EXTERNAL],
True)

View File

@ -29,6 +29,7 @@ import quantum
from quantum.common import exceptions from quantum.common import exceptions
from quantum.common import utils from quantum.common import utils
from quantum import context from quantum import context
from quantum.openstack.common import cfg
from quantum.openstack.common import importutils from quantum.openstack.common import importutils
from quantum.openstack.common import policy as common_policy from quantum.openstack.common import policy as common_policy
from quantum import policy from quantum import policy
@ -218,21 +219,21 @@ class QuantumPolicyTestCase(unittest.TestCase):
self.rules = { self.rules = {
"admin_or_network_owner": [["role:admin"], "admin_or_network_owner": [["role:admin"],
["tenant_id:%(network_tenant_id)s"]], ["tenant_id:%(network_tenant_id)s"]],
"admin_or_owner": [["role:admin"], ["tenant_id:%(tenant_id)s"]],
"admin_only": [["role:admin"]], "admin_only": [["role:admin"]],
"regular_user": [["role:user"]], "regular_user": [["role:user"]],
"shared": [["field:networks:shared=True"]],
"external": [["field:networks:router:external=True"]],
"default": [], "default": [],
"networks:private:read": [["rule:admin_only"]], "create_network": [["rule:admin_or_owner"]],
"networks:private:write": [["rule:admin_only"]],
"networks:shared:read": [["rule:regular_user"]],
"networks:shared:write": [["rule:admin_only"]],
"create_network": [],
"create_network:shared": [["rule:admin_only"]], "create_network:shared": [["rule:admin_only"]],
"update_network": [], "update_network": [],
"update_network:shared": [["rule:admin_only"]], "update_network:shared": [["rule:admin_only"]],
"get_network": [], "get_network": [["rule:admin_or_owner"],
["rule:shared"],
["rule:external"]],
"create_port:mac": [["rule:admin_or_network_owner"]], "create_port:mac": [["rule:admin_or_network_owner"]],
} }
@ -252,38 +253,38 @@ class QuantumPolicyTestCase(unittest.TestCase):
self.patcher.stop() self.patcher.stop()
policy.reset() policy.reset()
def test_nonadmin_write_on_private_returns_403(self): def _test_action_on_attr(self, context, action, attr, value,
action = "update_network" exception=None):
user_context = context.Context('', "user", roles=['user']) action = "%s_network" % action
# 384 is the int value of the bitmask for rw------ target = {'tenant_id': 'the_owner', attr: value}
target = {'tenant_id': 'the_owner', 'shared': False} if exception:
self.assertRaises(exceptions.PolicyNotAuthorized, policy.enforce, self.assertRaises(exception, policy.enforce,
user_context, action, target, None) context, action, target, None)
else:
def test_nonadmin_read_on_private_returns_403(self): result = policy.enforce(context, action, target, None)
action = "get_network"
user_context = context.Context('', "user", roles=['user'])
# 384 is the int value of the bitmask for rw------
target = {'tenant_id': 'the_owner', 'shared': False}
self.assertRaises(exceptions.PolicyNotAuthorized, policy.enforce,
user_context, action, target, None)
def test_nonadmin_write_on_shared_returns_403(self):
action = "update_network"
user_context = context.Context('', "user", roles=['user'])
# 384 is the int value of the bitmask for rw-r--r--
target = {'tenant_id': 'the_owner', 'shared': True}
self.assertRaises(exceptions.PolicyNotAuthorized, policy.enforce,
user_context, action, target, None)
def test_nonadmin_read_on_shared_returns_200(self):
action = "get_network"
user_context = context.Context('', "user", roles=['user'])
# 420 is the int value of the bitmask for rw-r--r--
target = {'tenant_id': 'the_owner', 'shared': True}
result = policy.enforce(user_context, action, target, None)
self.assertEqual(result, None) self.assertEqual(result, None)
def _test_nonadmin_action_on_attr(self, action, attr, value,
exception=None):
user_context = context.Context('', "user", roles=['user'])
self._test_action_on_attr(user_context, action, attr,
value, exception)
def test_nonadmin_write_on_private_fails(self):
self._test_nonadmin_action_on_attr('create', 'shared', False,
exceptions.PolicyNotAuthorized)
def test_nonadmin_read_on_private_fails(self):
self._test_nonadmin_action_on_attr('get', 'shared', False,
exceptions.PolicyNotAuthorized)
def test_nonadmin_write_on_shared_fails(self):
self._test_nonadmin_action_on_attr('create', 'shared', True,
exceptions.PolicyNotAuthorized)
def test_nonadmin_read_on_shared_succeeds(self):
self._test_nonadmin_action_on_attr('get', 'shared', True)
def _test_enforce_adminonly_attribute(self, action): def _test_enforce_adminonly_attribute(self, action):
admin_context = context.get_admin_context() admin_context = context.get_admin_context()
target = {'shared': True} target = {'shared': True}
@ -298,7 +299,7 @@ class QuantumPolicyTestCase(unittest.TestCase):
def test_enforce_adminoly_attribute_nonadminctx_returns_403(self): def test_enforce_adminoly_attribute_nonadminctx_returns_403(self):
action = "create_network" action = "create_network"
target = {'shared': True} target = {'shared': True, 'tenant_id': 'somebody_else'}
self.assertRaises(exceptions.PolicyNotAuthorized, policy.enforce, self.assertRaises(exceptions.PolicyNotAuthorized, policy.enforce,
self.context, action, target, None) self.context, action, target, None)