Remove policy.json from charm for ussuri
Glance (in ussuri) uses policy-in-code, and so that policy.json file doesn't ship with the package. This means that the charm can't rely on the file existing ussuri onwards. This patchset changes the way the charm uses policy.json by switching it to a charm determined policy.yaml file (preferred format) with the only 3 options that the charm determines to enforce. Also add yaml vars to focal-ussuri bundle This brings it into line with the other charms that are part of the the enable-focal topic. This makes it easier to add a new bundle just by changing a couple of variables. Closes-Bug: #1872996 Change-Id: I47f19272a4e0af3781843608b76304ce8ba1e2b8
This commit is contained in:
parent
a0f5d548f8
commit
848eb535a4
@ -71,6 +71,35 @@ class GlanceContext(OSContextGenerator):
|
|||||||
return ctxt
|
return ctxt
|
||||||
|
|
||||||
|
|
||||||
|
class GlancePolicyContext(OSContextGenerator):
|
||||||
|
"""This Context is only used from Ussuri onwards. At Ussuri, Glance
|
||||||
|
implemented policy-in-code, and thus didn't ship with a policy.json.
|
||||||
|
Therefore, the charm introduces a 'policy.yaml' file that is used to
|
||||||
|
provide the override here.
|
||||||
|
|
||||||
|
Note that this is separate from policy overrides as it's a charm config
|
||||||
|
option that has existed prior to its introduction.
|
||||||
|
|
||||||
|
Update *_image_location policy to restrict to admin role.
|
||||||
|
|
||||||
|
We do this unconditonally and keep a record of the original as installed by
|
||||||
|
the package.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __call__(self):
|
||||||
|
if config('restrict-image-location-operations'):
|
||||||
|
policy_value = 'role:admin'
|
||||||
|
else:
|
||||||
|
policy_value = ''
|
||||||
|
|
||||||
|
ctxt = {
|
||||||
|
"get_image_location": policy_value,
|
||||||
|
"set_image_location": policy_value,
|
||||||
|
"delete_image_location": policy_value,
|
||||||
|
}
|
||||||
|
return ctxt
|
||||||
|
|
||||||
|
|
||||||
class CephGlanceContext(OSContextGenerator):
|
class CephGlanceContext(OSContextGenerator):
|
||||||
interfaces = ['ceph-glance']
|
interfaces = ['ceph-glance']
|
||||||
|
|
||||||
|
@ -272,7 +272,7 @@ def object_store_joined():
|
|||||||
return
|
return
|
||||||
|
|
||||||
[image_service_joined(rid) for rid in relation_ids('image-service')]
|
[image_service_joined(rid) for rid in relation_ids('image-service')]
|
||||||
update_image_location_policy()
|
update_image_location_policy(CONFIGS)
|
||||||
CONFIGS.write(GLANCE_API_CONF)
|
CONFIGS.write(GLANCE_API_CONF)
|
||||||
|
|
||||||
|
|
||||||
@ -400,7 +400,7 @@ def config_changed():
|
|||||||
# NOTE(jamespage): trigger any configuration related changes
|
# NOTE(jamespage): trigger any configuration related changes
|
||||||
# for cephx permissions restrictions
|
# for cephx permissions restrictions
|
||||||
ceph_changed()
|
ceph_changed()
|
||||||
update_image_location_policy()
|
update_image_location_policy(CONFIGS)
|
||||||
|
|
||||||
# call the policy overrides handler which will install any policy overrides
|
# call the policy overrides handler which will install any policy overrides
|
||||||
maybe_do_policyd_overrides_on_config_changed(
|
maybe_do_policyd_overrides_on_config_changed(
|
||||||
@ -443,6 +443,9 @@ def upgrade_charm():
|
|||||||
reinstall_paste_ini(force_reinstall=packages_removed)
|
reinstall_paste_ini(force_reinstall=packages_removed)
|
||||||
configure_https()
|
configure_https()
|
||||||
update_nrpe_config()
|
update_nrpe_config()
|
||||||
|
# NOTE(ajkavanagh) the update_image_location_policy() call below isn't
|
||||||
|
# called with CONFIGS as the config files all get re-written after the
|
||||||
|
# call.
|
||||||
update_image_location_policy()
|
update_image_location_policy()
|
||||||
CONFIGS.write_all()
|
CONFIGS.write_all()
|
||||||
if packages_removed:
|
if packages_removed:
|
||||||
|
@ -36,6 +36,7 @@ from charmhelpers.core.hookenv import (
|
|||||||
config,
|
config,
|
||||||
log,
|
log,
|
||||||
INFO,
|
INFO,
|
||||||
|
WARNING,
|
||||||
relation_ids,
|
relation_ids,
|
||||||
service_name,
|
service_name,
|
||||||
)
|
)
|
||||||
@ -119,6 +120,11 @@ GLANCE_REGISTRY_PASTE = os.path.join(GLANCE_CONF_DIR,
|
|||||||
GLANCE_API_PASTE = os.path.join(GLANCE_CONF_DIR,
|
GLANCE_API_PASTE = os.path.join(GLANCE_CONF_DIR,
|
||||||
'glance-api-paste.ini')
|
'glance-api-paste.ini')
|
||||||
GLANCE_POLICY_FILE = os.path.join(GLANCE_CONF_DIR, "policy.json")
|
GLANCE_POLICY_FILE = os.path.join(GLANCE_CONF_DIR, "policy.json")
|
||||||
|
# NOTE(ajkavanagh): from Ussuri, glance switched to policy-in-code; this is the
|
||||||
|
# policy.yaml file (as there is not packaged policy.json or .yaml) that is used
|
||||||
|
# to provide the image_location override config value:
|
||||||
|
# 'restrict-image-location-operations'
|
||||||
|
GLANCE_POLICY_YAML = os.path.join(GLANCE_CONF_DIR, "policy.yaml")
|
||||||
CEPH_CONF = "/etc/ceph/ceph.conf"
|
CEPH_CONF = "/etc/ceph/ceph.conf"
|
||||||
CHARM_CEPH_CONF = '/var/lib/charm/{}/ceph.conf'
|
CHARM_CEPH_CONF = '/var/lib/charm/{}/ceph.conf'
|
||||||
|
|
||||||
@ -220,6 +226,10 @@ CONFIG_FILES = OrderedDict([
|
|||||||
'contexts': [],
|
'contexts': [],
|
||||||
'services': ['apache2'],
|
'services': ['apache2'],
|
||||||
}),
|
}),
|
||||||
|
(GLANCE_POLICY_YAML, {
|
||||||
|
'hook_contexts': [glance_contexts.GlancePolicyContext()],
|
||||||
|
'services': [],
|
||||||
|
}),
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
@ -270,8 +280,8 @@ def register_configs():
|
|||||||
CONFIG_FILES[GLANCE_SWIFT_CONF]['hook_contexts'])
|
CONFIG_FILES[GLANCE_SWIFT_CONF]['hook_contexts'])
|
||||||
|
|
||||||
if cmp_release >= 'ussuri':
|
if cmp_release >= 'ussuri':
|
||||||
configs.register(GLANCE_POLICY_FILE,
|
configs.register(GLANCE_POLICY_YAML,
|
||||||
CONFIG_FILES[GLANCE_POLICY_FILE]['hook_contexts'])
|
CONFIG_FILES[GLANCE_POLICY_YAML]['hook_contexts'])
|
||||||
|
|
||||||
return configs
|
return configs
|
||||||
|
|
||||||
@ -599,28 +609,66 @@ def is_api_ready(configs):
|
|||||||
return (not incomplete_relation_data(configs, REQUIRED_INTERFACES))
|
return (not incomplete_relation_data(configs, REQUIRED_INTERFACES))
|
||||||
|
|
||||||
|
|
||||||
def update_image_location_policy():
|
def update_image_location_policy(configs=None):
|
||||||
"""Update *_image_location policy to restrict to admin role.
|
"""Update *_image_location policy to restrict to admin role.
|
||||||
|
|
||||||
We do this unconditonally and keep a record of the original as installed by
|
We do this unconditonally and keep a record of the original as installed by
|
||||||
the package.
|
the package.
|
||||||
|
|
||||||
|
For ussuri, the charm updates/writes the policy.yaml file. The configs
|
||||||
|
param is optional as the caller may already be writing all the configs.
|
||||||
|
From ussuri onwards glance is policy-in-code (rather than using a
|
||||||
|
policy.json) and, therefore, policy files are essentially all overrides.
|
||||||
|
|
||||||
|
From ussuri, this function deletes the policy.json file and alternatively
|
||||||
|
writes the GLANCE_POLICY_YAML file via the configs object.
|
||||||
|
|
||||||
|
:param configs: The configs for the charm
|
||||||
|
:type configs: Optional[:class:templating.OSConfigRenderer()]
|
||||||
"""
|
"""
|
||||||
if CompareOpenStackReleases(os_release('glance-common')) < 'kilo':
|
_res = os_release('glance-common')
|
||||||
|
cmp = CompareOpenStackReleases(_res)
|
||||||
|
if cmp < 'kilo':
|
||||||
# NOTE(hopem): at the time of writing we are unable to do this for
|
# NOTE(hopem): at the time of writing we are unable to do this for
|
||||||
# earlier than Kilo due to LP: #1502136
|
# earlier than Kilo due to LP: #1502136
|
||||||
return
|
return
|
||||||
|
if cmp >= 'ussuri':
|
||||||
|
# If the policy.json exists, then remove it as it's the packaged
|
||||||
|
# version from a previous version of OpenStack, and thus not used.
|
||||||
|
if os.path.isfile(GLANCE_POLICY_FILE):
|
||||||
|
try:
|
||||||
|
os.remove(GLANCE_POLICY_FILE)
|
||||||
|
except Exception as e:
|
||||||
|
log("Problem removing file: {}: {}"
|
||||||
|
.format(GLANCE_POLICY_FILE, str(e)))
|
||||||
|
# if the caller supplied a configs param then update the
|
||||||
|
# GLANCE_POLICY_FILE using its context.
|
||||||
|
if configs is not None:
|
||||||
|
configs.write(GLANCE_POLICY_YAML)
|
||||||
|
return
|
||||||
|
|
||||||
|
# otherwise the OpenStack release after kilo and before ussuri, so continue
|
||||||
|
# modifying the existing policy.json file.
|
||||||
db = kv()
|
db = kv()
|
||||||
policies = ["get_image_location", "set_image_location",
|
policies = ["get_image_location", "set_image_location",
|
||||||
"delete_image_location"]
|
"delete_image_location"]
|
||||||
|
|
||||||
|
try:
|
||||||
|
with open(GLANCE_POLICY_FILE) as f:
|
||||||
|
pmap = json.load(f)
|
||||||
|
except IOError as e:
|
||||||
|
log("Problem opening glance policy file: {}. Error was:{}"
|
||||||
|
.format(GLANCE_POLICY_FILE, str(e)),
|
||||||
|
level=WARNING)
|
||||||
|
return
|
||||||
|
|
||||||
for policy_key in policies:
|
for policy_key in policies:
|
||||||
# Save original value at time of first install in case we ever need to
|
# Save original value at time of first install in case we ever need to
|
||||||
# revert.
|
# revert.
|
||||||
db_key = "policy_{}".format(policy_key)
|
db_key = "policy_{}".format(policy_key)
|
||||||
if db.get(db_key) is None:
|
if db.get(db_key) is None:
|
||||||
p = json.loads(open(GLANCE_POLICY_FILE).read())
|
if policy_key in pmap:
|
||||||
if policy_key in p:
|
db.set(db_key, pmap[policy_key])
|
||||||
db.set(db_key, p[policy_key])
|
|
||||||
db.flush()
|
db.flush()
|
||||||
else:
|
else:
|
||||||
log("key '{}' not found in policy file".format(policy_key),
|
log("key '{}' not found in policy file".format(policy_key),
|
||||||
@ -631,6 +679,9 @@ def update_image_location_policy():
|
|||||||
else:
|
else:
|
||||||
policy_value = ''
|
policy_value = ''
|
||||||
|
|
||||||
|
new_policies = {k: policy_value for k in policies}
|
||||||
|
for policy_key, policy_value in new_policies.items():
|
||||||
log("Updating Glance policy file setting policy "
|
log("Updating Glance policy file setting policy "
|
||||||
"'{}':'{}'".format(policy_key, policy_value), level=INFO)
|
"'{}': '{}'".format(policy_key, policy_value), level=INFO)
|
||||||
update_json_file(GLANCE_POLICY_FILE, {policy_key: policy_value})
|
|
||||||
|
update_json_file(GLANCE_POLICY_FILE, new_policies)
|
||||||
|
@ -1,5 +0,0 @@
|
|||||||
{
|
|
||||||
"delete_image_location": "rule:default",
|
|
||||||
"get_image_location": "rule:default",
|
|
||||||
"set_image_location": "rule:default"
|
|
||||||
}
|
|
5
templates/ussuri/policy.yaml
Normal file
5
templates/ussuri/policy.yaml
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# The file is controlled by Juju -- do not edit by hand as it will be
|
||||||
|
# overwritten
|
||||||
|
"get_image_location": "{{ get_image_location }}"
|
||||||
|
"set_image_location": "{{ set_image_location }}"
|
||||||
|
"delete_image_location": "{{ delete_image_location }}"
|
@ -1,3 +1,6 @@
|
|||||||
|
variables:
|
||||||
|
openstack-origin: &openstack-origin distro
|
||||||
|
|
||||||
series: focal
|
series: focal
|
||||||
|
|
||||||
comment:
|
comment:
|
||||||
@ -13,32 +16,48 @@ machines:
|
|||||||
'3':
|
'3':
|
||||||
'4':
|
'4':
|
||||||
|
|
||||||
relations:
|
|
||||||
- ["keystone:shared-db", "keystone-mysql-router:shared-db"]
|
|
||||||
- ["glance:shared-db", "glance-mysql-router:shared-db"]
|
|
||||||
- ["glance:identity-service", "keystone:identity-service"]
|
|
||||||
- ["glance-mysql-router:db-router", "mysql-innodb-cluster:db-router"]
|
|
||||||
- ["keystone-mysql-router:db-router", "mysql-innodb-cluster:db-router"]
|
|
||||||
|
|
||||||
applications:
|
applications:
|
||||||
|
|
||||||
glance-mysql-router:
|
glance-mysql-router:
|
||||||
charm: cs:~openstack-charmers-next/mysql-router
|
charm: cs:~openstack-charmers-next/mysql-router
|
||||||
keystone-mysql-router:
|
keystone-mysql-router:
|
||||||
charm: cs:~openstack-charmers-next/mysql-router
|
charm: cs:~openstack-charmers-next/mysql-router
|
||||||
|
|
||||||
mysql-innodb-cluster:
|
mysql-innodb-cluster:
|
||||||
charm: cs:~openstack-charmers-next/mysql-innodb-cluster
|
charm: cs:~openstack-charmers-next/mysql-innodb-cluster
|
||||||
num_units: 3
|
num_units: 3
|
||||||
|
options:
|
||||||
|
source: *openstack-origin
|
||||||
to:
|
to:
|
||||||
- '0'
|
- '0'
|
||||||
- '1'
|
- '1'
|
||||||
- '2'
|
- '2'
|
||||||
|
|
||||||
keystone:
|
keystone:
|
||||||
charm: cs:~openstack-charmers-next/keystone
|
charm: cs:~openstack-charmers-next/keystone
|
||||||
num_units: 1
|
num_units: 1
|
||||||
|
options:
|
||||||
|
openstack-origin: *openstack-origin
|
||||||
to:
|
to:
|
||||||
- '3'
|
- '3'
|
||||||
|
|
||||||
glance:
|
glance:
|
||||||
charm: ../../../glance
|
charm: ../../../glance
|
||||||
num_units: 1
|
num_units: 1
|
||||||
|
options:
|
||||||
|
openstack-origin: *openstack-origin
|
||||||
to:
|
to:
|
||||||
- '4'
|
- '4'
|
||||||
|
|
||||||
|
relations:
|
||||||
|
|
||||||
|
- - "keystone:shared-db"
|
||||||
|
- "keystone-mysql-router:shared-db"
|
||||||
|
- - "glance:shared-db"
|
||||||
|
- "glance-mysql-router:shared-db"
|
||||||
|
- - "glance:identity-service"
|
||||||
|
- "keystone:identity-service"
|
||||||
|
- - "glance-mysql-router:db-router"
|
||||||
|
- "mysql-innodb-cluster:db-router"
|
||||||
|
- - "keystone-mysql-router:db-router"
|
||||||
|
- "mysql-innodb-cluster:db-router"
|
||||||
|
@ -471,32 +471,30 @@ class TestGlanceUtils(CharmTestCase):
|
|||||||
utils.update_image_location_policy()
|
utils.update_image_location_policy()
|
||||||
self.assertFalse(mock_kv.called)
|
self.assertFalse(mock_kv.called)
|
||||||
|
|
||||||
|
# Function has no effect in ussuri onwards
|
||||||
|
mock_os_release.return_value = 'ussuri'
|
||||||
|
utils.update_image_location_policy()
|
||||||
|
self.assertFalse(mock_kv.called)
|
||||||
|
|
||||||
mock_os_release.return_value = 'kilo'
|
mock_os_release.return_value = 'kilo'
|
||||||
utils.update_image_location_policy()
|
utils.update_image_location_policy()
|
||||||
self.assertTrue(mock_kv.called)
|
self.assertTrue(mock_kv.called)
|
||||||
|
|
||||||
mock_update_json_file.assert_has_calls([
|
mock_update_json_file.assert_has_calls([
|
||||||
call('/etc/glance/policy.json',
|
call('/etc/glance/policy.json',
|
||||||
{'get_image_location': ''}),
|
{'get_image_location': '',
|
||||||
call('/etc/glance/policy.json',
|
'set_image_location': '',
|
||||||
{'set_image_location': ''}),
|
'delete_image_location': ''})])
|
||||||
call('/etc/glance/policy.json',
|
|
||||||
{'delete_image_location': ''})])
|
|
||||||
|
|
||||||
mock_update_json_file.reset_mock()
|
mock_update_json_file.reset_mock()
|
||||||
config['restrict-image-location-operations'] = True
|
config['restrict-image-location-operations'] = True
|
||||||
utils.update_image_location_policy()
|
utils.update_image_location_policy()
|
||||||
mock_update_json_file.assert_has_calls([
|
mock_update_json_file.assert_has_calls([
|
||||||
call('/etc/glance/policy.json',
|
call('/etc/glance/policy.json',
|
||||||
{'get_image_location': 'role:admin'}),
|
{'get_image_location': 'role:admin',
|
||||||
call('/etc/glance/policy.json',
|
'set_image_location': 'role:admin',
|
||||||
{'set_image_location': 'role:admin'}),
|
'delete_image_location': 'role:admin'})])
|
||||||
call('/etc/glance/policy.json',
|
|
||||||
{'delete_image_location': 'role:admin'})])
|
|
||||||
|
|
||||||
db_obj.get.assert_has_calls([call('policy_get_image_location'),
|
db_obj.get.assert_has_calls([call('policy_get_image_location'),
|
||||||
call('policy_set_image_location'),
|
call('policy_set_image_location'),
|
||||||
call('policy_delete_image_location')])
|
call('policy_delete_image_location')])
|
||||||
db_obj.set.assert_has_calls([call('policy_get_image_location', ''),
|
|
||||||
call('policy_set_image_location', ''),
|
|
||||||
call('policy_delete_image_location',
|
|
||||||
'')])
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user