Implement secure RBAC for secretstore API

Add new system scope specific RBAC rules for the secretstore API.
    The new rules allow all roles to list and get secret stores.

Change-Id: Ibb19e9854e8bafd2a454c0792503c6f4360e7cf7
This commit is contained in:
Ade Lee 2021-03-09 00:30:29 -05:00
parent 9a16e9114f
commit 060ca2ee36
4 changed files with 47 additions and 40 deletions

View File

@ -13,11 +13,13 @@
from oslo_policy import policy from oslo_policy import policy
_READER = "role:reader"
rules = [ rules = [
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name='secretstores:get', name='secretstores:get',
check_str='rule:admin', check_str=f'rule:all_users or {_READER}',
scope_types=[], scope_types=['project', 'system'],
description='Get list of available secret store backends.', description='Get list of available secret store backends.',
operations=[ operations=[
{ {
@ -28,8 +30,8 @@ rules = [
), ),
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name='secretstores:get_global_default', name='secretstores:get_global_default',
check_str='rule:admin', check_str=f'rule:all_users or {_READER}',
scope_types=[], scope_types=['project', 'system'],
description='Get a reference to the secret store that is used as ' + description='Get a reference to the secret store that is used as ' +
'default secret store backend for the deployment.', 'default secret store backend for the deployment.',
operations=[ operations=[
@ -41,8 +43,8 @@ rules = [
), ),
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name='secretstores:get_preferred', name='secretstores:get_preferred',
check_str='rule:admin', check_str=f'rule:all_users or {_READER}',
scope_types=[], scope_types=['project', 'system'],
description='Get a reference to the preferred secret store if ' + description='Get a reference to the preferred secret store if ' +
'assigned previously.', 'assigned previously.',
operations=[ operations=[
@ -55,7 +57,7 @@ rules = [
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name='secretstore_preferred:post', name='secretstore_preferred:post',
check_str='rule:admin', check_str='rule:admin',
scope_types=[], scope_types=['project'],
description='Set a secret store backend to be preferred store ' + description='Set a secret store backend to be preferred store ' +
'backend for their project.', 'backend for their project.',
operations=[ operations=[
@ -68,7 +70,7 @@ rules = [
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name='secretstore_preferred:delete', name='secretstore_preferred:delete',
check_str='rule:admin', check_str='rule:admin',
scope_types=[], scope_types=['project'],
description='Remove preferred secret store backend setting for ' + description='Remove preferred secret store backend setting for ' +
'their project.', 'their project.',
operations=[ operations=[
@ -80,8 +82,8 @@ rules = [
), ),
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name='secretstore:get', name='secretstore:get',
check_str='rule:admin', check_str=f'rule:all_users or {_READER}',
scope_types=[], scope_types=['project', 'system'],
description='Get details of secret store by its ID.', description='Get details of secret store by its ID.',
operations=[ operations=[
{ {

View File

@ -1164,28 +1164,19 @@ class WhenTestingSecretStoresResource(BaseTestCase):
self.assertIsNotNone(self.policy_enforcer.rules) self.assertIsNotNone(self.policy_enforcer.rules)
def test_should_pass_get_all_secret_stores(self): def test_should_pass_get_all_secret_stores(self):
self._assert_pass_rbac(['admin'], self._assert_pass_rbac(
self._invoke_on_get) ['admin', 'observer', 'audit', 'creator', 'reader'],
self._invoke_on_get)
def test_should_raise_get_all_secret_stores(self):
self._assert_fail_rbac([None, 'creator', 'observer', 'audit'],
self._invoke_on_get)
def test_should_pass_get_global_default(self): def test_should_pass_get_global_default(self):
self._assert_pass_rbac(['admin'], self._assert_pass_rbac(
self._invoke_get_global_default) ['admin', 'observer', 'audit', 'creator', 'reader'],
self._invoke_get_global_default)
def test_should_raise_get_global_default(self):
self._assert_fail_rbac([None, 'creator', 'observer', 'audit'],
self._invoke_get_global_default)
def test_should_pass_get_preferred(self): def test_should_pass_get_preferred(self):
self._assert_pass_rbac(['admin'], self._assert_pass_rbac(
self._invoke_get_preferred) ['admin', 'observer', 'audit', 'creator', 'reader'],
self._invoke_get_preferred)
def test_should_raise_get_preferred(self):
self._assert_fail_rbac([None, 'creator', 'observer', 'audit'],
self._invoke_get_preferred)
def _invoke_on_get(self): def _invoke_on_get(self):
self.resource.on_get(self.req, self.resp) self.resource.on_get(self.req, self.resp)
@ -1235,12 +1226,9 @@ class WhenTestingSecretStoreResource(BaseTestCase):
self.assertIsNotNone(self.policy_enforcer.rules) self.assertIsNotNone(self.policy_enforcer.rules)
def test_should_pass_get_a_secret_store(self): def test_should_pass_get_a_secret_store(self):
self._assert_pass_rbac(['admin'], self._assert_pass_rbac(
self._invoke_on_get) ['admin', 'observer', 'audit', 'creator', 'reader'],
self._invoke_on_get)
def test_should_raise_get_a_secret_store(self):
self._assert_fail_rbac([None, 'creator', 'observer', 'audit'],
self._invoke_on_get)
def _invoke_on_get(self): def _invoke_on_get(self):
self.resource.on_get(self.req, self.resp) self.resource.on_get(self.req, self.resp)
@ -1278,8 +1266,9 @@ class WhenTestingPreferredSecretStoreResource(BaseTestCase):
self._invoke_on_post) self._invoke_on_post)
def test_should_raise_set_preferred_secret_store(self): def test_should_raise_set_preferred_secret_store(self):
self._assert_fail_rbac([None, 'creator', 'observer', 'audit'], self._assert_fail_rbac(
self._invoke_on_post) [None, 'creator', 'observer', 'audit', 'reader'],
self._invoke_on_post)
def _invoke_on_post(self): def _invoke_on_post(self):
self.resource.on_post(self.req, self.resp) self.resource.on_post(self.req, self.resp)

View File

@ -29,6 +29,13 @@ admin_b = CONF.rbac_users.admin_b
observer_b = CONF.rbac_users.observer_b observer_b = CONF.rbac_users.observer_b
test_user_data_when_enabled = { test_user_data_when_enabled = {
'with_admin_a': {'user': admin_a, 'expected_return': 200},
'with_creator_a': {'user': creator_a, 'expected_return': 200},
'with_observer_a': {'user': observer_a, 'expected_return': 200},
'with_auditor_a': {'user': auditor_a, 'expected_return': 200},
}
test_user_data_admin_ops_when_enabled = {
'with_admin_a': {'user': admin_a, 'expected_return': 200}, 'with_admin_a': {'user': admin_a, 'expected_return': 200},
'with_creator_a': {'user': creator_a, 'expected_return': 403}, 'with_creator_a': {'user': creator_a, 'expected_return': 403},
'with_observer_a': {'user': observer_a, 'expected_return': 403}, 'with_observer_a': {'user': observer_a, 'expected_return': 403},
@ -37,9 +44,9 @@ test_user_data_when_enabled = {
test_user_data_when_not_enabled = { test_user_data_when_not_enabled = {
'with_admin_a': {'user': admin_a, 'expected_return': 404}, 'with_admin_a': {'user': admin_a, 'expected_return': 404},
'with_creator_a': {'user': creator_a, 'expected_return': 403}, 'with_creator_a': {'user': creator_a, 'expected_return': 404},
'with_observer_a': {'user': observer_a, 'expected_return': 403}, 'with_observer_a': {'user': observer_a, 'expected_return': 404},
'with_auditor_a': {'user': auditor_a, 'expected_return': 403}, 'with_auditor_a': {'user': auditor_a, 'expected_return': 404},
} }
@ -183,7 +190,7 @@ class SecretStoresTestCase(base.TestCase):
@testcase.skipUnless(base.conf_multiple_backends_enabled, 'executed only ' @testcase.skipUnless(base.conf_multiple_backends_enabled, 'executed only '
'when multiple backends support is enabled in ' 'when multiple backends support is enabled in '
'barbican server side') 'barbican server side')
@utils.parameterized_dataset(test_user_data_when_enabled) @utils.parameterized_dataset(test_user_data_admin_ops_when_enabled)
def test_unset_project_preferred_store_multiple_enabled(self, user, def test_unset_project_preferred_store_multiple_enabled(self, user,
expected_return): expected_return):

View File

@ -0,0 +1,9 @@
---
features:
- |
Implement secure-rbac for secretstores resource.
security:
- |
The current policy only allows users with the admin role to
list and get secretstore resources. The new policy allows all
users to perform these operations.