Merge "Remove endpoint_type from configuration"
This commit is contained in:
commit
fcc2088409
@ -8,7 +8,7 @@ Overview
|
|||||||
========
|
========
|
||||||
Ceph project is a powerful distributed storage system. It contains object store
|
Ceph project is a powerful distributed storage system. It contains object store
|
||||||
and provides a RADOS Gateway Swift API which is compatible with OpenStack Swift
|
and provides a RADOS Gateway Swift API which is compatible with OpenStack Swift
|
||||||
API. These two APIs use different formats for their temporary URLs.
|
API.
|
||||||
|
|
||||||
Ironic added support for RADOS Gateway temporary URL in the Mitaka release.
|
Ironic added support for RADOS Gateway temporary URL in the Mitaka release.
|
||||||
|
|
||||||
@ -17,56 +17,56 @@ Configure Ironic and Glance with RADOS Gateway
|
|||||||
|
|
||||||
#. Install Ceph storage with RADOS Gateway. See `Ceph documentation <http://docs.ceph.com/docs>`_.
|
#. Install Ceph storage with RADOS Gateway. See `Ceph documentation <http://docs.ceph.com/docs>`_.
|
||||||
|
|
||||||
#. Create RADOS Gateway credentials for Glance by executing the following
|
#. Configure RADOS Gateway to use keystone for authentication. See
|
||||||
commands on the RADOS Gateway admin host::
|
`Integrating with OpenStack Keystone <http://docs.ceph.com/docs/master/radosgw/keystone/>`_
|
||||||
|
|
||||||
sudo radosgw-admin user create --uid="GLANCE_USERNAME" --display-name="User for Glance"
|
#. Register RADOS Gateway endpoint in the keystone catalog, with the same
|
||||||
|
format swift uses, as the ``object-store`` service. URL example:
|
||||||
|
|
||||||
sudo radosgw-admin subuser create --uid=GLANCE_USERNAME --subuser=GLANCE_USERNAME:swift --access=full
|
``http://rados.example.com:8080/swift/v1/AUTH_$(project_id)s``.
|
||||||
|
|
||||||
sudo radosgw-admin key create --subuser=GLANCE_USERNAME:swift --key-type=swift --secret=STORE_KEY
|
In the ceph configuration, make sure radosgw is configured with the
|
||||||
|
following value::
|
||||||
|
|
||||||
sudo radosgw-admin user modify --uid=GLANCE_USERNAME --temp-url-key=TEMP_URL_KEY
|
rgw swift account in url = True
|
||||||
|
|
||||||
Replace GLANCE_USERNAME with a user name for Glance access, and replace
|
|
||||||
STORE_KEY and TEMP_URL_KEY with suitable keys.
|
|
||||||
|
|
||||||
Note: Do not use "--gen-secret" CLI parameter because it will cause the
|
|
||||||
"radosgw-admin" utility to generate keys with slash symbols which do not
|
|
||||||
work with Glance.
|
|
||||||
|
|
||||||
#. Configure Glance API service for RADOS Swift API as backend. Edit the
|
#. Configure Glance API service for RADOS Swift API as backend. Edit the
|
||||||
configuration file for the Glance API service (is typically located at
|
configuration file for the Glance API service (is typically located at
|
||||||
``/etc/glance/glance-api.conf``). Replace RADOS_IP and PORT with the IP/port
|
``/etc/glance/glance-api.conf``)::
|
||||||
of the RADOS Gateway API service::
|
|
||||||
|
|
||||||
[glance_store]
|
[glance_store]
|
||||||
|
|
||||||
stores = file, http, swift
|
stores = file, http, swift
|
||||||
default_store = swift
|
default_store = swift
|
||||||
swift_store_auth_version = 1
|
default_swift_reference=ref1
|
||||||
swift_store_auth_address = http://RADOS_IP:PORT/auth/1.0
|
swift_store_config_file=/etc/glance/glance-swift-creds.conf
|
||||||
swift_store_user = GLANCE_USERNAME:swift
|
|
||||||
swift_store_key = STORE_KEY
|
|
||||||
swift_store_container = glance
|
swift_store_container = glance
|
||||||
swift_store_create_container_on_put = True
|
swift_store_create_container_on_put = True
|
||||||
|
|
||||||
|
In the file referenced in ``swift_store_config_file`` option, add the
|
||||||
|
following::
|
||||||
|
|
||||||
|
[ref1]
|
||||||
|
user = <service project>:<service user name>
|
||||||
|
key = <service user password>
|
||||||
|
user_domain_id = default
|
||||||
|
project_domain_id = default
|
||||||
|
auth_version = 3
|
||||||
|
auth_address = http://keystone.example.com/identity
|
||||||
|
|
||||||
|
Values for user and key options correspond to keystone credentials for
|
||||||
|
RADOS Gateway service user.
|
||||||
|
|
||||||
Note: RADOS Gateway uses FastCGI protocol for interacting with HTTP server.
|
Note: RADOS Gateway uses FastCGI protocol for interacting with HTTP server.
|
||||||
Read your HTTP server documentation if you want to enable HTTPS support.
|
Read your HTTP server documentation if you want to enable HTTPS support.
|
||||||
|
|
||||||
#. Restart Glance API service and upload all needed images.
|
#. Restart Glance API service and upload all needed images.
|
||||||
|
|
||||||
#. Change Ironic configuration file on the conductor host(s) as follows::
|
#. If you're using custom container name in RADOS, change Ironic configuration
|
||||||
|
file on the conductor host(s) as follows::
|
||||||
|
|
||||||
[glance]
|
[glance]
|
||||||
|
|
||||||
swift_container = glance
|
swift_container = glance
|
||||||
swift_api_version = v1
|
|
||||||
swift_endpoint_url = http://RADOS_IP:PORT
|
|
||||||
swift_temp_url_key = TEMP_URL_KEY
|
|
||||||
|
|
||||||
[deploy]
|
|
||||||
|
|
||||||
object_store_endpoint_type = radosgw
|
|
||||||
|
|
||||||
#. Restart Ironic conductor service(s).
|
#. Restart Ironic conductor service(s).
|
||||||
|
@ -378,7 +378,6 @@ glance:
|
|||||||
swift_temp_url_duration = 3600
|
swift_temp_url_duration = 3600
|
||||||
swift_temp_url_expected_download_start_delay = 0
|
swift_temp_url_expected_download_start_delay = 0
|
||||||
swift_temp_url_key = ***
|
swift_temp_url_key = ***
|
||||||
temp_url_endpoint_type = swift
|
|
||||||
timeout = None
|
timeout = None
|
||||||
|
|
||||||
ilo:
|
ilo:
|
||||||
|
@ -61,8 +61,8 @@ Configuration drive storage in an object store
|
|||||||
|
|
||||||
Under normal circumstances, the configuration drive can be stored in the
|
Under normal circumstances, the configuration drive can be stored in the
|
||||||
Bare Metal service when the size is less than 64KB. Optionally, if the size
|
Bare Metal service when the size is less than 64KB. Optionally, if the size
|
||||||
is larger than 64KB there is support to store it in swift or radosgw backed
|
is larger than 64KB there is support to store it in a swift endpoint. Both
|
||||||
object store. Both swift and radosgw use swift-style APIs.
|
swift and radosgw use swift-style APIs.
|
||||||
|
|
||||||
The following option in ``/etc/ironic/ironic.conf`` enables swift as an object
|
The following option in ``/etc/ironic/ironic.conf`` enables swift as an object
|
||||||
store backend to store config drive. This uses the Identity service to
|
store backend to store config drive. This uses the Identity service to
|
||||||
@ -83,7 +83,6 @@ instead. ::
|
|||||||
...
|
...
|
||||||
|
|
||||||
configdrive_use_object_store = True
|
configdrive_use_object_store = True
|
||||||
object_store_endpoint_type = radosgw
|
|
||||||
|
|
||||||
[swift]
|
[swift]
|
||||||
...
|
...
|
||||||
|
@ -67,7 +67,6 @@ and Object Storage service as described below.
|
|||||||
|
|
||||||
[glance]
|
[glance]
|
||||||
|
|
||||||
temp_url_endpoint_type = swift
|
|
||||||
swift_endpoint_url = http://openstack/swift
|
swift_endpoint_url = http://openstack/swift
|
||||||
swift_account = AUTH_bc39f1d9dcf9486899088007789ae643
|
swift_account = AUTH_bc39f1d9dcf9486899088007789ae643
|
||||||
swift_container = glance
|
swift_container = glance
|
||||||
|
@ -149,40 +149,26 @@ class GlanceImageService(base_image_service.BaseImageService,
|
|||||||
endpoint_url = re.sub('/v1/AUTH_[^/]+/?$', '', endpoint_url)
|
endpoint_url = re.sub('/v1/AUTH_[^/]+/?$', '', endpoint_url)
|
||||||
|
|
||||||
key = CONF.glance.swift_temp_url_key
|
key = CONF.glance.swift_temp_url_key
|
||||||
if CONF.deploy.object_store_endpoint_type == 'radosgw':
|
account = CONF.glance.swift_account
|
||||||
chunks = urlparse.urlsplit(CONF.glance.swift_endpoint_url)
|
if not account:
|
||||||
if not chunks.path:
|
swift_session = swift.get_swift_session()
|
||||||
endpoint_url = urlparse.urljoin(
|
auth_ref = swift_session.auth.get_auth_ref(swift_session)
|
||||||
endpoint_url, 'swift')
|
account = 'AUTH_%s' % auth_ref.project_id
|
||||||
elif chunks.path != '/swift':
|
|
||||||
raise exc.InvalidParameterValue(
|
|
||||||
_('Swift endpoint URL should only contain scheme, '
|
|
||||||
'hostname, optional port and optional /swift path '
|
|
||||||
'without trailing slash; provided value is: %s')
|
|
||||||
% endpoint_url)
|
|
||||||
|
|
||||||
template = '/{api_version}/{container}/{object_id}'
|
if not key:
|
||||||
else:
|
swift_api = swift.SwiftAPI()
|
||||||
account = CONF.glance.swift_account
|
key_header = 'x-account-meta-temp-url-key'
|
||||||
if not account:
|
key = swift_api.connection.head_account().get(key_header)
|
||||||
swift_session = swift.get_swift_session()
|
|
||||||
auth_ref = swift_session.auth.get_auth_ref(swift_session)
|
|
||||||
account = 'AUTH_%s' % auth_ref.project_id
|
|
||||||
|
|
||||||
if not key:
|
if not key:
|
||||||
swift_api = swift.SwiftAPI()
|
raise exc.MissingParameterValue(_(
|
||||||
key_header = 'x-account-meta-temp-url-key'
|
'Swift temporary URLs require a shared secret to be '
|
||||||
key = swift_api.connection.head_account().get(key_header)
|
'created. You must provide "swift_temp_url_key" as a '
|
||||||
|
'config option or pre-generate the key on the project '
|
||||||
|
'used to access Swift.'))
|
||||||
|
|
||||||
if not key:
|
url_fragments['account'] = account
|
||||||
raise exc.MissingParameterValue(_(
|
template = '/{api_version}/{account}/{container}/{object_id}'
|
||||||
'Swift temporary URLs require a shared secret to be '
|
|
||||||
'created. You must provide "swift_temp_url_key" as a '
|
|
||||||
'config option or pre-generate the key on the project '
|
|
||||||
'used to access Swift.'))
|
|
||||||
|
|
||||||
url_fragments['account'] = account
|
|
||||||
template = '/{api_version}/{account}/{container}/{object_id}'
|
|
||||||
|
|
||||||
url_path = template.format(**url_fragments)
|
url_path = template.format(**url_fragments)
|
||||||
|
|
||||||
@ -197,11 +183,6 @@ class GlanceImageService(base_image_service.BaseImageService,
|
|||||||
|
|
||||||
def _validate_temp_url_config(self):
|
def _validate_temp_url_config(self):
|
||||||
"""Validate the required settings for a temporary URL."""
|
"""Validate the required settings for a temporary URL."""
|
||||||
if (not CONF.glance.swift_temp_url_key
|
|
||||||
and CONF.deploy.object_store_endpoint_type != 'swift'):
|
|
||||||
raise exc.MissingParameterValue(_(
|
|
||||||
'Swift temporary URLs require a shared secret to be created. '
|
|
||||||
'You must provide "swift_temp_url_key" as a config option.'))
|
|
||||||
if (CONF.glance.swift_temp_url_duration
|
if (CONF.glance.swift_temp_url_duration
|
||||||
< CONF.glance.swift_temp_url_expected_download_start_delay):
|
< CONF.glance.swift_temp_url_expected_download_start_delay):
|
||||||
raise exc.InvalidParameterValue(_(
|
raise exc.InvalidParameterValue(_(
|
||||||
|
@ -44,45 +44,39 @@ class SwiftAPI(object):
|
|||||||
"""Underlying Swift connection object."""
|
"""Underlying Swift connection object."""
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
"""Initialize the connection with swift or radosgw
|
"""Initialize the connection with swift
|
||||||
|
|
||||||
:raises: ConfigInvalid if required keystone authorization credentials
|
:raises: ConfigInvalid if required keystone authorization credentials
|
||||||
with swift are missing.
|
with swift are missing.
|
||||||
"""
|
"""
|
||||||
params = {'retries': CONF.swift.swift_max_retries}
|
params = {'retries': CONF.swift.swift_max_retries}
|
||||||
if CONF.deploy.object_store_endpoint_type == 'radosgw':
|
# NOTE(pas-ha) swiftclient still (as of 3.3.0) does not use
|
||||||
params.update({'authurl': CONF.swift.auth_url,
|
# (adapter-based) SessionClient, and uses the passed in session
|
||||||
'user': CONF.swift.username,
|
# only to resolve endpoint and get a token,
|
||||||
'key': CONF.swift.password})
|
# but not to make further requests to Swift itself (LP 1736135).
|
||||||
else:
|
# Thus we need to deconstruct back all the adapter- and
|
||||||
# NOTE(pas-ha) swiftclient still (as of 3.3.0) does not use
|
# session-related args as loaded by keystoneauth from config
|
||||||
# (adapter-based) SessionClient, and uses the passed in session
|
# to pass them to the client explicitly.
|
||||||
# only to resolve endpoint and get a token,
|
# TODO(pas-ha) re-write this when swiftclient is brought on par
|
||||||
# but not to make further requests to Swift itself (LP 1736135).
|
# with other OS clients re auth plugins, sessions and adapters
|
||||||
# Thus we need to deconstruct back all the adapter- and
|
# support.
|
||||||
# session-related args as loaded by keystoneauth from config
|
# TODO(pas-ha) pass the context here and use token from context
|
||||||
# to pass them to the client explicitly.
|
# with service auth
|
||||||
# TODO(pas-ha) re-write this when swiftclient is brought on par
|
params['session'] = session = get_swift_session()
|
||||||
# with other OS clients re auth plugins, sessions and adapters
|
adapter = keystone.get_adapter('swift', session=session)
|
||||||
# support.
|
params['os_options'] = {'object_storage_url': adapter.get_endpoint()}
|
||||||
# TODO(pas-ha) pass the context here and use token from context
|
# deconstruct back session-related options
|
||||||
# with service auth
|
params['timeout'] = session.timeout
|
||||||
params['session'] = session = get_swift_session()
|
if session.verify is False:
|
||||||
adapter = keystone.get_adapter('swift', session=session)
|
params['insecure'] = True
|
||||||
params['os_options'] = {
|
elif isinstance(session.verify, six.string_types):
|
||||||
'object_storage_url': adapter.get_endpoint()}
|
params['cacert'] = session.verify
|
||||||
# deconstruct back session-related options
|
if session.cert:
|
||||||
params['timeout'] = session.timeout
|
# NOTE(pas-ha) although setting cert as path to single file
|
||||||
if session.verify is False:
|
# with both client cert and key is supported by Session,
|
||||||
params['insecure'] = True
|
# keystoneauth loading always sets the session.cert
|
||||||
elif isinstance(session.verify, six.string_types):
|
# as tuple of cert and key.
|
||||||
params['cacert'] = session.verify
|
params['cert'], params['cert_key'] = session.cert
|
||||||
if session.cert:
|
|
||||||
# NOTE(pas-ha) although setting cert as path to single file
|
|
||||||
# with both client cert and key is supported by Session,
|
|
||||||
# keystoneauth loading always sets the session.cert
|
|
||||||
# as tuple of cert and key.
|
|
||||||
params['cert'], params['cert_key'] = session.cert
|
|
||||||
|
|
||||||
self.connection = swift_client.Connection(**params)
|
self.connection = swift_client.Connection(**params)
|
||||||
|
|
||||||
|
@ -86,8 +86,6 @@ class BaseConductorManager(object):
|
|||||||
:raises: DriverLoadError if an enabled driver cannot be loaded.
|
:raises: DriverLoadError if an enabled driver cannot be loaded.
|
||||||
:raises: DriverNameConflict if a classic driver and a dynamic driver
|
:raises: DriverNameConflict if a classic driver and a dynamic driver
|
||||||
are both enabled and have the same name.
|
are both enabled and have the same name.
|
||||||
:raises: ConfigInvalid if required config options for connection with
|
|
||||||
radosgw are missing while storing config drive.
|
|
||||||
"""
|
"""
|
||||||
if self._started:
|
if self._started:
|
||||||
raise RuntimeError(_('Attempt to start an already running '
|
raise RuntimeError(_('Attempt to start an already running '
|
||||||
@ -139,18 +137,6 @@ class BaseConductorManager(object):
|
|||||||
|
|
||||||
self._collect_periodic_tasks(admin_context)
|
self._collect_periodic_tasks(admin_context)
|
||||||
|
|
||||||
# Check for required config options if object_store_endpoint_type is
|
|
||||||
# radosgw
|
|
||||||
if (CONF.deploy.configdrive_use_object_store
|
|
||||||
and CONF.deploy.object_store_endpoint_type == "radosgw"):
|
|
||||||
if (None in (CONF.swift.auth_url, CONF.swift.username,
|
|
||||||
CONF.swift.password)):
|
|
||||||
msg = _("Parameters missing to make a connection with "
|
|
||||||
"radosgw. Ensure that [swift]/auth_url, "
|
|
||||||
"[swift]/username, and [swift]/password are all "
|
|
||||||
"configured.")
|
|
||||||
raise exception.ConfigInvalid(msg)
|
|
||||||
|
|
||||||
# clear all target_power_state with locks by this conductor
|
# clear all target_power_state with locks by this conductor
|
||||||
self.dbapi.clear_node_target_power_state(self.host)
|
self.dbapi.clear_node_target_power_state(self.host)
|
||||||
# clear all locks held by this conductor before registering
|
# clear all locks held by this conductor before registering
|
||||||
|
@ -3340,14 +3340,14 @@ def _get_configdrive_obj_name(node):
|
|||||||
def _store_configdrive(node, configdrive):
|
def _store_configdrive(node, configdrive):
|
||||||
"""Handle the storage of the config drive.
|
"""Handle the storage of the config drive.
|
||||||
|
|
||||||
If configured, the config drive data are uploaded to swift or radosgw.
|
If configured, the config drive data are uploaded to a swift endpoint.
|
||||||
The Node's instance_info is updated to include either the temporary
|
The Node's instance_info is updated to include either the temporary
|
||||||
Swift URL from the upload, or if no upload, the actual config drive data.
|
Swift URL from the upload, or if no upload, the actual config drive data.
|
||||||
|
|
||||||
:param node: an Ironic node object.
|
:param node: an Ironic node object.
|
||||||
:param configdrive: A gzipped and base64 encoded configdrive.
|
:param configdrive: A gzipped and base64 encoded configdrive.
|
||||||
:raises: SwiftOperationError if an error occur when uploading the
|
:raises: SwiftOperationError if an error occur when uploading the
|
||||||
config drive to swift or radosgw.
|
config drive to the swift endpoint.
|
||||||
:raises: ConfigInvalid if required keystone authorization credentials
|
:raises: ConfigInvalid if required keystone authorization credentials
|
||||||
with swift are missing.
|
with swift are missing.
|
||||||
|
|
||||||
|
@ -95,15 +95,7 @@ opts = [
|
|||||||
deprecated_name='configdrive_use_swift',
|
deprecated_name='configdrive_use_swift',
|
||||||
help=_('Whether to upload the config drive to object store. '
|
help=_('Whether to upload the config drive to object store. '
|
||||||
'Set this option to True to store config drive '
|
'Set this option to True to store config drive '
|
||||||
'in swift or radosgw.')),
|
'in a swift endpoint.')),
|
||||||
cfg.StrOpt('object_store_endpoint_type',
|
|
||||||
default='swift',
|
|
||||||
deprecated_group='glance',
|
|
||||||
deprecated_name='temp_url_endpoint_type',
|
|
||||||
choices=[('swift', _('use Object Storage service')),
|
|
||||||
('radosgw', _('use RADOS object store'))],
|
|
||||||
help=_('Type of object store endpoint type to be '
|
|
||||||
'used as a backend')),
|
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@ -68,7 +68,7 @@ opts = [
|
|||||||
'swift_endpoint_url',
|
'swift_endpoint_url',
|
||||||
help=_('The "endpoint" (scheme, hostname, optional port) for '
|
help=_('The "endpoint" (scheme, hostname, optional port) for '
|
||||||
'the Swift URL of the form '
|
'the Swift URL of the form '
|
||||||
'"endpoint_url/api_version/[account/]container/object_id". '
|
'"endpoint_url/api_version/account/container/object_id". '
|
||||||
'Do not include trailing "/". '
|
'Do not include trailing "/". '
|
||||||
'For example, use "https://swift.example.com". If using RADOS '
|
'For example, use "https://swift.example.com". If using RADOS '
|
||||||
'Gateway, endpoint may also contain /swift path; if it does '
|
'Gateway, endpoint may also contain /swift path; if it does '
|
||||||
@ -79,7 +79,7 @@ opts = [
|
|||||||
default='v1',
|
default='v1',
|
||||||
help=_('The Swift API version to create a temporary URL for. '
|
help=_('The Swift API version to create a temporary URL for. '
|
||||||
'Defaults to "v1". Swift temporary URL format: '
|
'Defaults to "v1". Swift temporary URL format: '
|
||||||
'"endpoint_url/api_version/[account/]container/object_id"')),
|
'"endpoint_url/api_version/account/container/object_id"')),
|
||||||
cfg.StrOpt(
|
cfg.StrOpt(
|
||||||
'swift_account',
|
'swift_account',
|
||||||
help=_('The account that Glance uses to communicate with '
|
help=_('The account that Glance uses to communicate with '
|
||||||
@ -89,7 +89,7 @@ opts = [
|
|||||||
'If not set, the default value is calculated based on the ID '
|
'If not set, the default value is calculated based on the ID '
|
||||||
'of the project used to access Swift (as set in the [swift] '
|
'of the project used to access Swift (as set in the [swift] '
|
||||||
'section). Swift temporary URL format: '
|
'section). Swift temporary URL format: '
|
||||||
'"endpoint_url/api_version/[account/]container/object_id"')),
|
'"endpoint_url/api_version/account/container/object_id"')),
|
||||||
cfg.StrOpt(
|
cfg.StrOpt(
|
||||||
'swift_container',
|
'swift_container',
|
||||||
default='glance',
|
default='glance',
|
||||||
@ -97,7 +97,7 @@ opts = [
|
|||||||
'images in. Defaults to "glance", which is the default '
|
'images in. Defaults to "glance", which is the default '
|
||||||
'in glance-api.conf. '
|
'in glance-api.conf. '
|
||||||
'Swift temporary URL format: '
|
'Swift temporary URL format: '
|
||||||
'"endpoint_url/api_version/[account/]container/object_id"')),
|
'"endpoint_url/api_version/account/container/object_id"')),
|
||||||
cfg.IntOpt('swift_store_multiple_containers_seed',
|
cfg.IntOpt('swift_store_multiple_containers_seed',
|
||||||
default=0,
|
default=0,
|
||||||
help=_('This should match a config by the same name in the '
|
help=_('This should match a config by the same name in the '
|
||||||
|
@ -23,7 +23,6 @@ from keystoneauth1 import loading as kaloading
|
|||||||
import mock
|
import mock
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
from oslo_utils import uuidutils
|
from oslo_utils import uuidutils
|
||||||
from six.moves.urllib import parse as urlparse
|
|
||||||
import testtools
|
import testtools
|
||||||
|
|
||||||
from ironic.common import context
|
from ironic.common import context
|
||||||
@ -685,65 +684,6 @@ class TestGlanceSwiftTempURL(base.TestCase):
|
|||||||
self.service.swift_temp_url, image_info)
|
self.service.swift_temp_url, image_info)
|
||||||
self.assertFalse(tempurl_mock.called)
|
self.assertFalse(tempurl_mock.called)
|
||||||
|
|
||||||
@mock.patch('swiftclient.utils.generate_temp_url', autospec=True)
|
|
||||||
def test_swift_temp_url_radosgw(self, tempurl_mock):
|
|
||||||
self.config(object_store_endpoint_type='radosgw', group='deploy')
|
|
||||||
path = ('/v1'
|
|
||||||
'/glance'
|
|
||||||
'/757274c4-2856-4bd2-bb20-9a4a231e187b')
|
|
||||||
tempurl_mock.return_value = (
|
|
||||||
path + '?temp_url_sig=hmacsig&temp_url_expires=1400001200')
|
|
||||||
|
|
||||||
self.service._validate_temp_url_config = mock.Mock()
|
|
||||||
|
|
||||||
temp_url = self.service.swift_temp_url(image_info=self.fake_image)
|
|
||||||
|
|
||||||
self.assertEqual(
|
|
||||||
(urlparse.urljoin(CONF.glance.swift_endpoint_url, 'swift')
|
|
||||||
+ tempurl_mock.return_value),
|
|
||||||
temp_url)
|
|
||||||
tempurl_mock.assert_called_with(
|
|
||||||
path=path,
|
|
||||||
seconds=CONF.glance.swift_temp_url_duration,
|
|
||||||
key=CONF.glance.swift_temp_url_key,
|
|
||||||
method='GET')
|
|
||||||
|
|
||||||
@mock.patch('swiftclient.utils.generate_temp_url', autospec=True)
|
|
||||||
def test_swift_temp_url_radosgw_endpoint_with_swift(self, tempurl_mock):
|
|
||||||
self.config(swift_endpoint_url='https://swift.radosgw.com/swift',
|
|
||||||
group='glance')
|
|
||||||
self.config(object_store_endpoint_type='radosgw', group='deploy')
|
|
||||||
path = ('/v1'
|
|
||||||
'/glance'
|
|
||||||
'/757274c4-2856-4bd2-bb20-9a4a231e187b')
|
|
||||||
tempurl_mock.return_value = (
|
|
||||||
path + '?temp_url_sig=hmacsig&temp_url_expires=1400001200')
|
|
||||||
|
|
||||||
self.service._validate_temp_url_config = mock.Mock()
|
|
||||||
|
|
||||||
temp_url = self.service.swift_temp_url(image_info=self.fake_image)
|
|
||||||
|
|
||||||
self.assertEqual(
|
|
||||||
CONF.glance.swift_endpoint_url + tempurl_mock.return_value,
|
|
||||||
temp_url)
|
|
||||||
tempurl_mock.assert_called_with(
|
|
||||||
path=path,
|
|
||||||
seconds=CONF.glance.swift_temp_url_duration,
|
|
||||||
key=CONF.glance.swift_temp_url_key,
|
|
||||||
method='GET')
|
|
||||||
|
|
||||||
@mock.patch('swiftclient.utils.generate_temp_url', autospec=True)
|
|
||||||
def test_swift_temp_url_radosgw_endpoint_invalid(self, tempurl_mock):
|
|
||||||
self.config(swift_endpoint_url='https://swift.radosgw.com/eggs/',
|
|
||||||
group='glance')
|
|
||||||
self.config(object_store_endpoint_type='radosgw', group='deploy')
|
|
||||||
self.service._validate_temp_url_config = mock.Mock()
|
|
||||||
|
|
||||||
self.assertRaises(exception.InvalidParameterValue,
|
|
||||||
self.service.swift_temp_url,
|
|
||||||
self.fake_image)
|
|
||||||
self.assertFalse(tempurl_mock.called)
|
|
||||||
|
|
||||||
@mock.patch('swiftclient.utils.generate_temp_url', autospec=True)
|
@mock.patch('swiftclient.utils.generate_temp_url', autospec=True)
|
||||||
def test_swift_temp_url_multiple_containers(self, tempurl_mock):
|
def test_swift_temp_url_multiple_containers(self, tempurl_mock):
|
||||||
|
|
||||||
@ -777,20 +717,8 @@ class TestGlanceSwiftTempURL(base.TestCase):
|
|||||||
def test__validate_temp_url_config(self):
|
def test__validate_temp_url_config(self):
|
||||||
self.service._validate_temp_url_config()
|
self.service._validate_temp_url_config()
|
||||||
|
|
||||||
def test__validate_temp_url_key_no_exception(self):
|
def test__validate_temp_url_no_key_no_exception(self):
|
||||||
self.config(swift_temp_url_key=None, group='glance')
|
self.config(swift_temp_url_key=None, group='glance')
|
||||||
self.config(object_store_endpoint_type='swift', group='deploy')
|
|
||||||
self.service._validate_temp_url_config()
|
|
||||||
|
|
||||||
def test__validate_temp_url_key_exception(self):
|
|
||||||
self.config(swift_temp_url_key=None, group='glance')
|
|
||||||
self.config(object_store_endpoint_type='radosgw', group='deploy')
|
|
||||||
self.assertRaises(exception.MissingParameterValue,
|
|
||||||
self.service._validate_temp_url_config)
|
|
||||||
|
|
||||||
def test__validate_temp_url_no_account_exception_radosgw(self):
|
|
||||||
self.config(swift_account=None, group='glance')
|
|
||||||
self.config(object_store_endpoint_type='radosgw', group='deploy')
|
|
||||||
self.service._validate_temp_url_config()
|
self.service._validate_temp_url_config()
|
||||||
|
|
||||||
def test__validate_temp_url_endpoint_less_than_download_delay(self):
|
def test__validate_temp_url_endpoint_less_than_download_delay(self):
|
||||||
|
@ -57,30 +57,6 @@ class SwiftTestCase(base.TestCase):
|
|||||||
os_options={'object_storage_url': 'http://example.com/objects'}
|
os_options={'object_storage_url': 'http://example.com/objects'}
|
||||||
)
|
)
|
||||||
|
|
||||||
def test___init___radosgw(self, connection_mock, swift_session_mock):
|
|
||||||
"""Check if client is properly initialized with radosgw"""
|
|
||||||
|
|
||||||
auth_url = 'http://1.2.3.4'
|
|
||||||
username = 'foo'
|
|
||||||
password = 'foo_password'
|
|
||||||
CONF.set_override('object_store_endpoint_type', 'radosgw',
|
|
||||||
group='deploy')
|
|
||||||
opts = [cfg.StrOpt('auth_url'), cfg.StrOpt('username'),
|
|
||||||
cfg.StrOpt('password')]
|
|
||||||
CONF.register_opts(opts, group='swift')
|
|
||||||
|
|
||||||
CONF.set_override('auth_url', auth_url, group='swift')
|
|
||||||
CONF.set_override('username', username, group='swift')
|
|
||||||
CONF.set_override('password', password, group='swift')
|
|
||||||
|
|
||||||
swift.SwiftAPI()
|
|
||||||
params = {'authurl': auth_url,
|
|
||||||
'user': username,
|
|
||||||
'key': password,
|
|
||||||
'retries': 2}
|
|
||||||
connection_mock.assert_called_once_with(**params)
|
|
||||||
self.assertFalse(swift_session_mock.called)
|
|
||||||
|
|
||||||
@mock.patch.object(__builtin__, 'open', autospec=True)
|
@mock.patch.object(__builtin__, 'open', autospec=True)
|
||||||
def test_create_object(self, open_mock, connection_mock, keystone_mock):
|
def test_create_object(self, open_mock, connection_mock, keystone_mock):
|
||||||
swiftapi = swift.SwiftAPI()
|
swiftapi = swift.SwiftAPI()
|
||||||
|
@ -218,28 +218,6 @@ class StartStopTestCase(mgr_utils.ServiceSetUpMixin, db_base.DbTestCase):
|
|||||||
self.service.del_host()
|
self.service.del_host()
|
||||||
self.assertTrue(wait_mock.called)
|
self.assertTrue(wait_mock.called)
|
||||||
|
|
||||||
def test_start_fails_on_missing_config_for_configdrive(self):
|
|
||||||
"""Check to fail conductor on missing config options"""
|
|
||||||
|
|
||||||
missing_parameters_error = ("Parameters missing to make a "
|
|
||||||
"connection with radosgw")
|
|
||||||
CONF.set_override('configdrive_use_object_store', True,
|
|
||||||
group='deploy')
|
|
||||||
CONF.set_override('object_store_endpoint_type', 'radosgw',
|
|
||||||
group='deploy')
|
|
||||||
params = {'auth_url': 'http://1.2.3.4',
|
|
||||||
'username': 'foo', 'password': 'foo_pass'}
|
|
||||||
CONF.register_opts((cfg.StrOpt(x) for x in params),
|
|
||||||
group='swift')
|
|
||||||
for key, value in params.items():
|
|
||||||
test_params = params.copy()
|
|
||||||
test_params[key] = None
|
|
||||||
for test_key, test_value in test_params.items():
|
|
||||||
CONF.set_override(key, test_value, group='swift')
|
|
||||||
with self.assertRaisesRegex(exception.ConfigInvalid,
|
|
||||||
missing_parameters_error):
|
|
||||||
self._start_service()
|
|
||||||
|
|
||||||
def test_conductor_shutdown_flag(self):
|
def test_conductor_shutdown_flag(self):
|
||||||
self._start_service()
|
self._start_service()
|
||||||
self.assertFalse(self.service._shutdown)
|
self.assertFalse(self.service._shutdown)
|
||||||
|
@ -0,0 +1,9 @@
|
|||||||
|
---
|
||||||
|
upgrade:
|
||||||
|
- |
|
||||||
|
The ``swift/endpoint_type`` configuration option is now removed.
|
||||||
|
python-swiftclient 3.2.0 (Ocata) and above removed support for the native
|
||||||
|
URL type used by radosgw. Since using a ``swift/endpoint_type`` value of
|
||||||
|
``radosgw`` would fail anyway, it is removed. Deployers must now configure
|
||||||
|
ceph with ``rgw swift account in url = True``. This must be set before
|
||||||
|
upgrading to this release.
|
Loading…
x
Reference in New Issue
Block a user