diff --git a/etc/policy.json b/etc/policy.json index 9207142582e..4aab8d5190d 100644 --- a/etc/policy.json +++ b/etc/policy.json @@ -56,7 +56,9 @@ "update_network:router:external": "rule:admin_only", "delete_network": "rule:admin_or_owner", + "network_device": "field:port:device_owner=~^network:", "create_port": "", + "create_port:device_owner": "not rule:network_device or rule:admin_or_network_owner or rule:context_is_advsvc", "create_port:mac_address": "rule:admin_or_network_owner or rule:context_is_advsvc", "create_port:fixed_ips": "rule:admin_or_network_owner or rule:context_is_advsvc", "create_port:port_security_enabled": "rule:admin_or_network_owner or rule:context_is_advsvc", @@ -71,6 +73,7 @@ "get_port:binding:host_id": "rule:admin_only", "get_port:binding:profile": "rule:admin_only", "update_port": "rule:admin_or_owner or rule:context_is_advsvc", + "update_port:device_owner": "not rule:network_device or rule:admin_or_network_owner or rule:context_is_advsvc", "update_port:mac_address": "rule:admin_only or rule:context_is_advsvc", "update_port:fixed_ips": "rule:admin_or_network_owner or rule:context_is_advsvc", "update_port:port_security_enabled": "rule:admin_or_network_owner or rule:context_is_advsvc", diff --git a/neutron/api/v2/attributes.py b/neutron/api/v2/attributes.py index c2ffe7c462b..56beac9f459 100644 --- a/neutron/api/v2/attributes.py +++ b/neutron/api/v2/attributes.py @@ -731,7 +731,7 @@ RESOURCE_ATTRIBUTE_MAP = { 'is_visible': True}, 'device_owner': {'allow_post': True, 'allow_put': True, 'validate': {'type:string': DEVICE_OWNER_MAX_LEN}, - 'default': '', + 'default': '', 'enforce_policy': True, 'is_visible': True}, 'tenant_id': {'allow_post': True, 'allow_put': False, 'validate': {'type:string': TENANT_ID_MAX_LEN}, diff --git a/neutron/policy.py b/neutron/policy.py index e1d955a6022..99286a02661 100644 --- a/neutron/policy.py +++ b/neutron/policy.py @@ -290,6 +290,7 @@ class FieldCheck(policy.Check): self.field = field self.value = conv_func(value) + self.regex = re.compile(value[1:]) if value.startswith('~') else None def __call__(self, target_dict, cred_dict, enforcer): target_value = target_dict.get(self.field) @@ -299,6 +300,8 @@ class FieldCheck(policy.Check): "%(target_dict)s", {'field': self.field, 'target_dict': target_dict}) return False + if self.regex: + return bool(self.regex.match(target_value)) return target_value == self.value diff --git a/neutron/tests/etc/policy.json b/neutron/tests/etc/policy.json index 9207142582e..4aab8d5190d 100644 --- a/neutron/tests/etc/policy.json +++ b/neutron/tests/etc/policy.json @@ -56,7 +56,9 @@ "update_network:router:external": "rule:admin_only", "delete_network": "rule:admin_or_owner", + "network_device": "field:port:device_owner=~^network:", "create_port": "", + "create_port:device_owner": "not rule:network_device or rule:admin_or_network_owner or rule:context_is_advsvc", "create_port:mac_address": "rule:admin_or_network_owner or rule:context_is_advsvc", "create_port:fixed_ips": "rule:admin_or_network_owner or rule:context_is_advsvc", "create_port:port_security_enabled": "rule:admin_or_network_owner or rule:context_is_advsvc", @@ -71,6 +73,7 @@ "get_port:binding:host_id": "rule:admin_only", "get_port:binding:profile": "rule:admin_only", "update_port": "rule:admin_or_owner or rule:context_is_advsvc", + "update_port:device_owner": "not rule:network_device or rule:admin_or_network_owner or rule:context_is_advsvc", "update_port:mac_address": "rule:admin_only or rule:context_is_advsvc", "update_port:fixed_ips": "rule:admin_or_network_owner or rule:context_is_advsvc", "update_port:port_security_enabled": "rule:admin_or_network_owner or rule:context_is_advsvc", diff --git a/neutron/tests/unit/test_policy.py b/neutron/tests/unit/test_policy.py index 145005b600f..ed230179fdf 100644 --- a/neutron/tests/unit/test_policy.py +++ b/neutron/tests/unit/test_policy.py @@ -241,6 +241,7 @@ class NeutronPolicyTestCase(base.BaseTestCase): "regular_user": "role:user", "shared": "field:networks:shared=True", "external": "field:networks:router:external=True", + "network_device": "field:port:device_owner=~^network:", "default": '@', "create_network": "rule:admin_or_owner", @@ -252,6 +253,7 @@ class NeutronPolicyTestCase(base.BaseTestCase): "create_subnet": "rule:admin_or_network_owner", "create_port:mac": "rule:admin_or_network_owner or " "rule:context_is_advsvc", + "create_port:device_owner": "not rule:network_device", "update_port": "rule:admin_or_owner or rule:context_is_advsvc", "get_port": "rule:admin_or_owner or rule:context_is_advsvc", "delete_port": "rule:admin_or_owner or rule:context_is_advsvc", @@ -330,6 +332,20 @@ class NeutronPolicyTestCase(base.BaseTestCase): self._test_nonadmin_action_on_attr('create', 'shared', True, oslo_policy.PolicyNotAuthorized) + def test_create_port_device_owner_regex(self): + blocked_values = ('network:', 'network:abdef', 'network:dhcp', + 'network:router_interface') + for val in blocked_values: + self._test_advsvc_action_on_attr( + 'create', 'port', 'device_owner', val, + oslo_policy.PolicyNotAuthorized + ) + ok_values = ('network', 'networks', 'my_network:test', 'my_network:') + for val in ok_values: + self._test_advsvc_action_on_attr( + 'create', 'port', 'device_owner', val + ) + def test_advsvc_get_network_works(self): self._test_advsvc_action_on_attr('get', 'network', 'shared', False)