Refactor Freezer authentication with keystone

Supports v2 and v3 of keystone and using keystoneauth library for
authentication with sessions. Supports new OS environment variables

Change-Id: I5cf8a5460b81c071f5982c1a7b7750d1c861a86f
Implements: blueprint refactor-os-clients
This commit is contained in:
Saad Zaher 2016-04-28 10:57:08 +00:00
parent d0711f2a2a
commit 1b83ba7aef
11 changed files with 388 additions and 292 deletions

View File

@ -27,7 +27,6 @@ from oslo_log import log
from freezer.common import config as freezer_config from freezer.common import config as freezer_config
from freezer.engine.tar import tar_engine from freezer.engine.tar import tar_engine
from freezer import job from freezer import job
from freezer.openstack import openstack
from freezer.openstack import osclients from freezer.openstack import osclients
from freezer.storage import local from freezer.storage import local
from freezer.storage import multiple from freezer.storage import multiple
@ -58,23 +57,21 @@ def freezer_main(backup_args):
validator.validate(backup_args) validator.validate(backup_args)
work_dir = backup_args.work_dir work_dir = backup_args.work_dir
os_identity = backup_args.os_identity_api_version
max_segment_size = backup_args.max_segment_size max_segment_size = backup_args.max_segment_size
if backup_args.storage == 'swift' or (backup_args.__dict__['backup_media'] if backup_args.storage == 'swift' or (
in ['nova', 'cinder', 'cindernative']): backup_args.backup_media in ['nova', 'cinder', 'cindernative']):
client_manager = get_client_manager(backup_args.__dict__) backup_args.client_manager = get_client_manager(backup_args.__dict__)
if backup_args.storages: if backup_args.storages:
storage = multiple.MultipleStorage( storage = multiple.MultipleStorage(
work_dir, work_dir,
[storage_from_dict(x, work_dir, max_segment_size, os_identity) [storage_from_dict(x, work_dir, max_segment_size)
for x in backup_args.storages]) for x in backup_args.storages])
else: else:
storage = storage_from_dict(backup_args.__dict__, work_dir, storage = storage_from_dict(backup_args.__dict__, work_dir,
max_segment_size, os_identity) max_segment_size)
backup_args.__dict__['engine'] = tar_engine.TarBackupEngine( backup_args.engine = tar_engine.TarBackupEngine(
backup_args.compression, backup_args.compression,
backup_args.dereference_symlink, backup_args.dereference_symlink,
backup_args.exclude, backup_args.exclude,
@ -146,26 +143,24 @@ def parse_osrc(file_name):
with open(file_name, 'r') as osrc_file: with open(file_name, 'r') as osrc_file:
return config.osrc_parse(osrc_file.read()) return config.osrc_parse(osrc_file.read())
def get_client_manager(backup_args, os_identity_api_version=None):
def get_client_manager(backup_args):
if "osrc" in backup_args: if "osrc" in backup_args:
options = openstack.OpenstackOptions.create_from_dict( options = osclients.OpenstackOpts.create_from_dict(
parse_osrc(backup_args['osrc'])) parse_osrc(backup_args['osrc']))
else: else:
options = openstack.OpenstackOptions.create_from_env() options = osclients.OpenstackOpts.create_from_env().get_opts_dicts()
identity_api_version = (os_identity_api_version or
options.identity_api_version)
client_manager = osclients.ClientManager(
options=options,
insecure=backup_args.get('insecure') or False,
swift_auth_version=identity_api_version,
dry_run=backup_args.get('dry_run') or False)
backup_args['client_manager'] = client_manager client_manager = osclients.OSClientManager(
auth_url=options.pop('auth_url', None),
auth_method=options.pop('auth_method', 'password'),
dry_run=backup_args.get('dry_run', None),
**options
)
return client_manager return client_manager
def storage_from_dict(backup_args, work_dir, max_segment_size, def storage_from_dict(backup_args, work_dir, max_segment_size):
os_identity_api_version=None):
storage_name = backup_args['storage'] storage_name = backup_args['storage']
container = backup_args['container'] container = backup_args['container']
if storage_name == "swift": if storage_name == "swift":
@ -188,7 +183,7 @@ def storage_from_dict(backup_args, work_dir, max_segment_size,
def main(): def main():
"""freezer-agent/freezerc binary main execution""" """freezer-agent binary main execution"""
backup_args = None backup_args = None
try: try:
freezer_config.config() freezer_config.config()
@ -203,5 +198,4 @@ def main():
return fail(1, err, quiet) return fail(1, err, quiet)
if __name__ == '__main__': if __name__ == '__main__':
sys.exit(main()) sys.exit(main())

View File

@ -28,7 +28,7 @@ logging = log.getLogger(__name__)
home = os.path.expanduser("~") home = os.path.expanduser("~")
class BackupOs: class BackupOs(object):
def __init__(self, client_manager, container, storage): def __init__(self, client_manager, container, storage):
""" """
@ -76,8 +76,8 @@ class BackupOs:
headers = {"x-object-meta-name": instance._info['name'], headers = {"x-object-meta-name": instance._info['name'],
"x-object-meta-flavor-id": instance._info['flavor']['id']} "x-object-meta-flavor-id": instance._info['flavor']['id']}
self.storage.add_stream(stream, package, headers) self.storage.add_stream(stream, package, headers)
logging.info("[*] Deleting temporary image") logging.info("[*] Deleting temporary image {0}".format(image))
glance.images.delete(image) glance.images.delete(image.id)
def backup_cinder_by_glance(self, volume_id): def backup_cinder_by_glance(self, volume_id):
""" """

View File

@ -1,81 +0,0 @@
"""
(c) Copyright 2015,2016 Hewlett-Packard Development Company, L.P.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""
import os
class OpenstackOptions:
"""
Stores credentials for OpenStack API.
Can be created using
>> create_from_env()
or
>> create_from_dict(dict)
"""
def __init__(self, user_name, tenant_name, project_name, auth_url,
password, identity_api_version, tenant_id=None,
region_name=None, endpoint_type=None, cert=None,
insecure=False, verify=True):
self.user_name = user_name
self.tenant_name = tenant_name
self.auth_url = auth_url
self.password = password
self.tenant_id = tenant_id
self.project_name = project_name
self.identity_api_version = identity_api_version
self.region_name = region_name
self.endpoint_type = endpoint_type
self.cert = cert
self.insecure = insecure
self.verify = verify
if not (self.password and self.user_name and self.auth_url and
(self.tenant_name or self.project_name)):
raise Exception("Please set up in your env:"
"OS_USERNAME, OS_TENANT_NAME/OS_PROJECT_NAME,"
" OS_AUTH_URL, OS_PASSWORD")
@property
def os_options(self):
"""
:return: The OpenStack options which can have tenant_id,
auth_token, service_type, endpoint_type, tenant_name,
object_storage_url, region_name
"""
return {'tenant_id': self.tenant_id,
'tenant_name': self.tenant_name,
'project_name': self.project_name,
'identity_api_version': self.identity_api_version,
'region_name': self.region_name,
'endpoint_type': self.endpoint_type}
@staticmethod
def create_from_env():
return OpenstackOptions.create_from_dict(os.environ)
@staticmethod
def create_from_dict(src_dict):
return OpenstackOptions(
user_name=src_dict.get('OS_USERNAME', None),
tenant_name=src_dict.get('OS_TENANT_NAME', None),
project_name=src_dict.get('OS_PROJECT_NAME', None),
auth_url=src_dict.get('OS_AUTH_URL', None),
identity_api_version=src_dict.get('OS_IDENTITY_API_VERSION',
'2.0'),
password=src_dict.get('OS_PASSWORD', None),
tenant_id=src_dict.get('OS_TENANT_ID', None),
region_name=src_dict.get('OS_REGION_NAME', None),
endpoint_type=src_dict.get('OS_ENDPOINT_TYPE', None),
cert=src_dict.get('OS_CERT', None)
)

View File

@ -1,4 +1,5 @@
# (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. # (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
# (c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); # Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License. # you may not use this file except in compliance with the License.
@ -12,14 +13,17 @@
# See the License for the specific language governing permissions and # See the License for the specific language governing permissions and
# limitations under the License. # limitations under the License.
import os
import swiftclient
import time import time
from cinderclient import client as cclient from cinderclient.client import Client as cinder_client
from glanceclient import client as gclient from glanceclient.client import Client as glance_client
from novaclient import client as nclient from keystoneauth1 import loading
from keystoneauth1 import session
from novaclient.client import Client as nova_client
from oslo_config import cfg from oslo_config import cfg
from oslo_log import log from oslo_log import log
import swiftclient
from freezer.utils import utils from freezer.utils import utils
@ -27,158 +31,156 @@ CONF = cfg.CONF
logging = log.getLogger(__name__) logging = log.getLogger(__name__)
class ClientManager: class OSClientManager(object):
""" def __init__(self, auth_url, auth_method='password', **kwargs):
:type swift: swiftclient.Connection
:type glance: glanceclient.v1.client.Client
:type nova: novaclient.v2.client.Client
:type cinder: cinderclient.v1.client.Client
"""
def __init__(self, options, insecure=True,
swift_auth_version=2, dry_run=False):
"""
Creates manager of connections to swift, nova, glance and cinder
:param options: OpenstackOptions
:type options: freezer.openstack.openstack.OpenstackOptions
:param insecure:
:param swift_auth_version:
:param dry_run:
:return:
"""
self.options = options
self.insecure = insecure
self.swift_auth_version = swift_auth_version
self.dry_run = dry_run
self.cinder = None
self.swift = None self.swift = None
self.glance = None self.glance = None
self.nova = None self.nova = None
self.cinder = None
self.dry_run = kwargs.pop('dry_run', None)
loader = loading.get_plugin_loader(auth_method)
# copy the args for swift authentication !
self.swift_args = kwargs.copy()
self.swift_args['auth_url'] = auth_url
# client specific arguments !
self.client_kwargs = {}
# session specific arguments
session_kwargs = {}
if 'verify' in kwargs.keys():
session_kwargs['verify'] = kwargs.pop('verify')
if 'cacert' in kwargs.keys():
session_kwargs['cert'] = kwargs.pop('cacert')
# client specific args
if 'insecure' in kwargs.keys():
self.client_kwargs['insecure'] = kwargs.pop('insecure')
session_kwargs['verify'] = False
if 'region_name' in kwargs.keys():
self.client_kwargs['region_name'] = kwargs.pop('region_name')
if 'endpoint_type' in kwargs.keys():
self.client_kwargs['endpoint_type'] = kwargs.pop('endpoint_type')
if 'identity_api_version' in kwargs.keys():
kwargs.pop('identity_api_version')
if 'auth_version' in kwargs.keys():
kwargs.pop('auth_version')
if 'interface' in kwargs.keys():
self.client_kwargs['interface'] = kwargs.pop('interface')
def get_cinder(self): self.compute_version = kwargs.pop('compute_api_version', 2)
""" self.image_version = kwargs.pop('image_api_version', 2)
:rtype cinderclient.v1.client.Client self.volume_version = kwargs.pop('volume_api_version', 2)
:return: self.auth = loader.load_from_options(auth_url=auth_url, **kwargs)
"""
if not self.cinder:
self.create_cinder()
return self.cinder
def get_swift(self): self.sess = session.Session(auth=self.auth, **session_kwargs)
"""
:rtype swiftclient.Connection
:return: instance of swift client
"""
if not self.swift:
self.create_swift()
return self.swift
def get_glance(self): def create_nova(self):
""" """
:rtype glanceclient.v1.client.Client Use pre-initialized session to create an instance of nova client.
:return: :return: novaclient instance
""" """
if not self.glance: self.nova = nova_client(self.compute_version, session=self.sess,
self.create_glance() **self.client_kwargs)
return self.glance
def get_nova(self):
"""
:rtype
:return:
"""
if not self.nova:
self.create_nova()
return self.nova return self.nova
def create_glance(self):
"""
Use pre-initialized session to create an instance of glance client.
:return: glanceclient instance
"""
if 'endpoint_type' in self.client_kwargs.keys():
self.client_kwargs.pop('endpoint_type')
if 'insecure' in self.client_kwargs.keys():
self.client_kwargs.pop('insecure')
self.glance = glance_client(self.image_version, session=self.sess,
**self.client_kwargs)
return self.glance
def create_cinder(self): def create_cinder(self):
""" """
Creates client for cinder and caches it Use pre-initialized session to create an instance of cinder client.
:rtype cinderclient.v1.client.Client :return: cinderclient instance
:return: instance of cinder client
""" """
options = self.options self.cinder = cinder_client(self.volume_version, session=self.sess,
logging.info("[*] Creation of cinder client") **self.client_kwargs)
self.cinder = cclient.Client(
version="2",
username=options.user_name,
api_key=options.password,
project_id=options.tenant_name,
auth_url=options.auth_url,
region_name=options.region_name,
insecure=self.insecure,
endpoint_type=options.endpoint_type or 'publicURL',
service_type="volume",
cacert=options.cert)
return self.cinder return self.cinder
def create_swift(self): def create_swift(self):
""" """
Creates client for swift and caches it Swift client needs to be treated differently so we need to copy the
:rtype swiftclient.Connection arguments and provide it to swiftclient the correct way !
:return: instance of swift client :return: swiftclient instance
""" """
options = self.options os_options = {}
logging.info("[*] Creation of swift client") auth_version = None
if 'region_name' in self.swift_args.keys():
os_options['region_name'] = self.swift_args.get('region_name')
if 'endpoint_type' in self.swift_args.keys():
os_options['endpoint_type'] = self.swift_args.pop('endpoint_type')
if 'tenant_id' in self.swift_args.keys():
os_options['tenant_id'] = self.swift_args.pop('tenant_id')
if 'identity_api_version' in self.swift_args.keys():
os_options['identity_api_version'] = \
self.swift_args.pop('identity_api_version')
auth_version = os_options['identity_api_version']
if 'token' in self.swift_args.keys():
os_options['auth_token'] = self.swift_args.pop('token')
if 'auth_version' in self.swift_args.keys():
auth_version = self.swift_args.get('auth_version')
tenant_name = self.swift_args.get('project_name') or self.swift_args.\
get('tenant_name')
self.swift = swiftclient.client.Connection( self.swift = swiftclient.client.Connection(
authurl=options.auth_url, authurl=self.swift_args.get('auth_url'),
user=options.user_name, key=options.password, user=self.swift_args.get('username'),
tenant_name=options.tenant_name, key=self.swift_args.get('password'),
os_options=options.os_options, tenant_name=tenant_name,
auth_version=self.swift_auth_version, insecure=self.swift_args.get('insecure', False),
insecure=self.insecure, retries=6, cacert=self.swift_args.get('cacert', None),
cacert=options.cert) os_options=os_options,
auth_version=auth_version
)
if self.dry_run: if self.dry_run:
self.swift = DryRunSwiftclientConnectionWrapper(self.swift) self.swift = DryRunSwiftclientConnectionWrapper(self.swift)
return self.swift return self.swift
def create_glance(self): def get_nova(self):
""" """
Creates client for glance and caches it Get novaclient instance
:rtype glanceclient.v1.client.Client :return: novaclient instance
:return: instance of glance client
""" """
if not self.nova:
self.nova = self.create_nova()
return self.nova
from glanceclient.shell import OpenStackImagesShell def get_glance(self):
"""
options = self.options Get glanceclient instance
:return: glanceclient instance
logging.info("[*] Creation of glance client") """
if not self.glance:
self.glance = OpenStackImagesShell()._get_versioned_client('1', self.glance = self.create_glance()
utils.Bunch(os_username=options.user_name,
os_password=options.password,
os_tenant_name=options.tenant_name,
os_project_name=options.project_name,
os_auth_url=options.auth_url,
os_region_name=options.region_name,
endpoint_type=options.endpoint_type,
force_auth=False,
cacert=options.cert))
return self.glance return self.glance
def create_nova(self): def get_cinder(self):
""" """
Creates client for nova and caches it Get cinderclient instance
:return: :return: cinderclient instance
""" """
options = self.options if not self.cinder:
logging.info("[*] Creation of nova client") self.cinder = self.create_cinder()
return self.cinder
self.nova = nclient.Client( def get_swift(self):
version='2', """
username=options.user_name, Get swiftclient instance
api_key=options.password, :return: swiftclient instance
project_id=options.tenant_name, """
auth_url=options.auth_url, if not self.swift:
region_name=options.region_name, self.swift = self.create_swift()
insecure=self.insecure, return self.swift
cacert=options.cert)
return self.nova
def provide_snapshot(self, volume, snapshot_name): def provide_snapshot(self, volume, snapshot_name):
""" """
@ -275,7 +277,188 @@ class ClientManager:
return utils.ReSizeStream(stream, image.size, 1000000) return utils.ReSizeStream(stream, image.size, 1000000)
class DryRunSwiftclientConnectionWrapper: class OpenstackOpts(object):
"""
Gathering and maintaining the right Openstack credentials that will be used
to authenticate against keystone. Now we support keystone v2 and v3.
We need to provide a correct url that ends with either v2.0 or v3 or provide
auth_version or identity_api_version
"""
def __init__(self, auth_url, auth_method='password', auth_version=None,
username=None, password=None, region_name=None, cacert=None,
identity_api_version=None, project_id=None, project_name=None,
tenant_id=None, tenant_name=None, token=None, insecure=False,
endpoint_type='internalURL', interface=None,
compute_api_version=2, image_api_version=2,
volume_api_version=2, user_domain_name=None, domain_id=None,
user_domain_id=None, project_domain_id=None, domain_name=None,
project_domain_name=None):
"""
Authentication Options to build a valid opts dict to be used to
authenticate against keystone. You must provide auth_url with a vaild
Openstack version at the end v2.0 or v3 or provide auth_version.
:param auth_url: string Keystone API URL
:param auth_method: string defaults to password or token (not tested)
:param auth_version: string Keystone API version. 2.0 or 3
:param username: string A valid Username
:param password: string A valid Password
:param region_name: string Region name or None
:param cacert: string Path to CA certificate
:param identity_api_version: string Keystone API version to use
:param project_id: UUID string Project ID
:param project_name: string Project Name
:param tenant_id: string Project/ Tenant ID. Use with keystone v2.0 only
:param tenant_name: string Project/ Tenant Name. keystone v2.0 only
:param token: string Valid token. Only if auth_method is token
:param insecure: boolean Use insecure connections
:param endpoint_type: string publicURL, adminURL, internalURL
:param interface: string internal, ...
:param compute_api_version: int NOVA API version to use default 2
:param image_api_version: int Glance API version, default 2
:param volume_api_version: int Cinder API version, default 2
:param user_domain_name: string User Domain Name. only with keystone v3
:param domain_id: string Domain ID. Only with keystone v3
:param user_domain_id: string User Domain ID. only with keystone v3
:param project_domain_id: string Project Domain ID. keystone v3 only
:param domain_name: string Domain Name. only with keystone v3
:param project_domain_name: string Project Domain Name. keystone v3 only
:return: None
"""
self.auth_url = auth_url
self.auth_method = auth_method
self.auth_version = auth_version
self.username = username
self.password = password
self.region_name = region_name
self.cacert = cacert
self.identity_api_version = identity_api_version
self.tenant_id = tenant_id or project_id
self.project_id = project_id or tenant_id
self.project_name = project_name or tenant_name
self.tenant_name = tenant_name or project_name
self.token = token
self.insecure = insecure
self.endpoint_type = endpoint_type
self.interface = interface
self.compute_api_version = compute_api_version
self.image_api_version = image_api_version
self.volume_api_version = volume_api_version
self.user_domain_id = user_domain_id
self.user_domain_name = user_domain_name
self.project_domain_id = project_domain_id
self.project_domain_name = project_domain_name
self.domain_id = domain_id
self.domain_name = domain_name
if auth_url is None:
raise Exception('auth_url required to authenticate. Make sure to '
'export OS_AUTH_URL=http://keystone_url:5000/v3')
if auth_version is None and identity_api_version is None:
version = auth_url.rstrip('/').rsplit('/')[-1]
if version == 'v3':
self.auth_version = self.identity_api_version = str('3')
elif version == 'v2.0':
self.auth_version = self.identity_api_version = str('2.0')
else:
raise Exception('Keystone Auth version {0} is not supported!. '
'Generated from auth_url: {1}'.format(version,
auth_url))
logging.info('Authenticating with Keystone version: {0}, auth_url: {1},'
' username: {2}, project: {3}'.format(self.auth_version,
self.auth_url,
self.username,
self.project_name))
def get_opts_dicts(self):
"""
Return opentack auth arguments as dict
detects the auth version from url if not provided
handles certificate issues
"""
opts = self.__dict__
if self.auth_method == 'password':
opts.pop('token', None)
elif self.auth_method == 'token':
opts.pop('username', None)
opts.pop('password', None)
if not self.cacert:
opts['verify'] = False
opts['insecure'] = True
self.auth_version = str(self.auth_version)
self.identity_api_version = str(self.identity_api_version)
if self.auth_version == '3' or self.identity_api_version == '3':
opts['auth_version'] = opts['identity_api_version'] = '3'
opts.pop('tenant_id', None)
opts.pop('tenant_name', None)
elif self.auth_version in ['2.0', '2'] or self.identity_api_version in \
['2.0', '2']:
opts['auth_version'] = opts['identity_api_version'] = '2.0'
# these parameters won't work with keystone v2.0
opts.pop('project_id', None)
opts.pop('project_name', None)
opts.pop('project_domain_id', None)
opts.pop('project_domain_name', None)
opts.pop('user_domain_id', None)
opts.pop('user_domain_name', None)
opts.pop('domain_id', None)
opts.pop('domain_name', None)
else:
raise Exception('Keystone Auth version {0} is not supported!. '
'Generated from auth_url: {1}'.
format(self.auth_version, self.auth_url))
for i in opts.copy().keys():
if opts.get(i) is None:
opts.pop(i)
return opts
@staticmethod
def create_from_env():
"""
Parse environment variables and load Openstack related options.
:return:
"""
return OpenstackOpts.create_from_dict(os.environ)
@staticmethod
def create_from_dict(src_dict):
"""
Load Openstack arguments from dict and return OpenstackOpts object with
the correct parameters to authenticate.
:param src_dict: dict
:return: OpenstackOpts object with the passed arguments in place
"""
return OpenstackOpts(
auth_url=src_dict.get('OS_AUTH_URL'),
auth_method=src_dict.get('OS_AUTH_METHOD', 'password'),
auth_version=src_dict.get('OS_AUTH_VERSION', None),
username=src_dict.get('OS_USERNAME', None),
password=src_dict.get('OS_PASSWORD', None),
tenant_id=src_dict.get('OS_TENANT_ID', None),
tenant_name=src_dict.get('OS_TENANT_NAME', None),
project_id=src_dict.get('OS_PROJECT_ID', None),
project_name=src_dict.get('OS_PROJECT_NAME', None),
region_name=src_dict.get('OS_REGION_NAME', None),
endpoint_type=src_dict.get('OS_ENDPOINT_TYPE', 'publicURL'),
cacert=src_dict.get('OS_CACERT', None),
identity_api_version=src_dict.get('OS_IDENTITY_API_VERSION', None),
insecure=src_dict.get('OS_INSECURE', False),
token=src_dict.get('OS_TOKEN', None),
interface=src_dict.get('OS_INTERFACE', None),
user_domain_name=src_dict.get('OS_USER_DOMAIN_NAME', None),
user_domain_id=src_dict.get('OS_USER_DOMAIN_ID', None),
project_domain_id=src_dict.get('OS_PROJECT_DOMAIN_ID', None),
project_domain_name=src_dict.get('OS_PROJECT_DOMAIN_NAME', None),
domain_id=src_dict.get('OS_DOMAIN_ID'),
domain_name=src_dict.get('OS_DOMAIN_NAME'),
compute_api_version=src_dict.get('OS_COMPUTE_API_VERSION', 2),
volume_api_version=src_dict.get('OS_VOLUME_API_VERSION', 2),
image_api_version=src_dict.get('OS_IMAGE_API_VERSION', 2)
)
class DryRunSwiftclientConnectionWrapper(object):
def __init__(self, sw_connector): def __init__(self, sw_connector):
self.sw_connector = sw_connector self.sw_connector = sw_connector
self.get_object = sw_connector.get_object self.get_object = sw_connector.get_object

View File

@ -1,6 +1,7 @@
#!/usr/bin/env python #!/usr/bin/env python
# (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. # (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
# (c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P
# #
# Licensed under the Apache License, Version 2.0 (the "License"); # Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License. # you may not use this file except in compliance with the License.
@ -16,6 +17,7 @@
from mock import MagicMock from mock import MagicMock
from mock import Mock
import swiftclient import swiftclient
import multiprocessing import multiprocessing
@ -27,13 +29,14 @@ import re
from glanceclient.common.utils import IterableWithLength from glanceclient.common.utils import IterableWithLength
from freezer.storage import swift from freezer.storage import swift
from freezer.utils import utils from freezer.utils import utils
from freezer.openstack import openstack from freezer.openstack.osclients import OpenstackOpts
from freezer.openstack.osclients import OSClientManager
from freezer.engine.tar import tar_engine from freezer.engine.tar import tar_engine
os.environ['OS_REGION_NAME'] = 'testregion' os.environ['OS_REGION_NAME'] = 'testregion'
os.environ['OS_TENANT_ID'] = '0123456789' os.environ['OS_TENANT_ID'] = '0123456789'
os.environ['OS_PASSWORD'] = 'testpassword' os.environ['OS_PASSWORD'] = 'testpassword'
os.environ['OS_AUTH_URL'] = 'testauthurl' os.environ['OS_AUTH_URL'] = 'http://testauthurl/v2.0'
os.environ['OS_USERNAME'] = 'testusername' os.environ['OS_USERNAME'] = 'testusername'
os.environ['OS_TENANT_NAME'] = 'testtenantename' os.environ['OS_TENANT_NAME'] = 'testtenantename'
@ -316,10 +319,14 @@ class BackupOpt1:
self.cindernative_vol_id = '' self.cindernative_vol_id = ''
self.nova_inst_id = '' self.nova_inst_id = ''
self.lvm_snapperm = 'ro' self.lvm_snapperm = 'ro'
self.options = openstack.OpenstackOptions.create_from_dict(os.environ)
from freezer.openstack.osclients import ClientManager self.compression = 'gzip'
from mock import Mock self.storage = MagicMock()
self.client_manager = ClientManager(None, False, 2, False) self.engine = MagicMock()
opts = OpenstackOpts.create_from_env().get_opts_dicts()
self.client_manager = OSClientManager(opts.pop('auth_url'),
opts.pop('auth_method'),
**opts)
self.client_manager.get_swift = Mock( self.client_manager.get_swift = Mock(
return_value=FakeSwiftClient().client.Connection()) return_value=FakeSwiftClient().client.Connection())
self.client_manager.create_swift = self.client_manager.get_swift self.client_manager.create_swift = self.client_manager.get_swift
@ -327,8 +334,6 @@ class BackupOpt1:
self.container, self.container,
self.work_dir, self.work_dir,
self.max_segment_size) self.max_segment_size)
self.compression = 'gzip'
self.engine = tar_engine.TarBackupEngine( self.engine = tar_engine.TarBackupEngine(
self.compression, self.dereference_symlink, self.compression, self.dereference_symlink,
self.exclude, self.storage, False) self.exclude, self.storage, False)
@ -337,6 +342,7 @@ class BackupOpt1:
nova_client = MagicMock() nova_client = MagicMock()
self.client_manager.get_nova = Mock(return_value=nova_client) self.client_manager.get_nova = Mock(return_value=nova_client)
self.command = None self.command = None
@ -387,7 +393,7 @@ class Os:
@classmethod @classmethod
def exists(cls, directory=True): def exists(cls, directory=True):
return True return True
@classmethod @classmethod
def notexists(cls, directory=True): def notexists(cls, directory=True):
return False return False

View File

@ -1,6 +1,6 @@
""" """
(c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
(c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P.
Licensed under the Apache License, Version 2.0 (the "License"); Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. you may not use this file except in compliance with the License.
You may obtain a copy of the License at You may obtain a copy of the License at
@ -103,4 +103,4 @@ class QueuedThread(threading.Thread):
super(QueuedThread, self).run() super(QueuedThread, self).run()
except Exception as e: except Exception as e:
self.rich_queue.force_stop() self.rich_queue.force_stop()
raise e raise e

View File

@ -14,6 +14,7 @@ oslo.utils>=3.5.0 # Apache-2.0
oslo.i18n>=2.1.0 # Apache-2.0 oslo.i18n>=2.1.0 # Apache-2.0
oslo.log>=1.14.0 # Apache-2.0 oslo.log>=1.14.0 # Apache-2.0
oslo.config>=3.9.0 # Apache-2.0 oslo.config>=3.9.0 # Apache-2.0
keystoneauth1>=2.1.0 # Apache-2.0
PyMySQL>=0.6.2 # MIT License PyMySQL>=0.6.2 # MIT License
pymongo!=3.1,>=3.0.2 # Apache-2.0 pymongo!=3.1,>=3.0.2 # Apache-2.0

View File

@ -1,4 +1,5 @@
# (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. # (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
# (c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); # Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License. # you may not use this file except in compliance with the License.
@ -12,61 +13,46 @@
# See the License for the specific language governing permissions and # See the License for the specific language governing permissions and
# limitations under the License. # limitations under the License.
import unittest import unittest
import mock import mock
from freezer.openstack import openstack
from freezer.openstack import osclients from freezer.openstack import osclients
from freezer.utils import utils
class TestOsClients(unittest.TestCase): class TestOsClients(unittest.TestCase):
def setUp(self):
fake_options = openstack.OpenstackOptions( self.opts = osclients.OpenstackOpts(
user_name="user", tenant_name="tenant", project_name="project", username="user", tenant_name="tenant", project_name="project",
auth_url="url", password="password", identity_api_version="3", auth_url="url/v3", password="password", identity_api_version="3",
insecure=False, cert='cert', verify=True) insecure=False, cacert='cert', user_domain_name='Default',
project_domain_name='Default').get_opts_dicts()
self.client_manager = osclients.OSClientManager(auth_method=self.opts.pop('auth_method'),
auth_url=self.opts.pop('auth_url'),
**self.opts)
def test_init(self): def test_init(self):
osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.get_cinder()
def test_create_cinder(self): def test_create_cinder(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.create_cinder()
client.create_cinder()
def test_create_swift(self): def test_create_swift(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.create_swift()
client.create_swift()
def test_create_nova(self): def test_create_nova(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.create_nova()
client.create_nova()
def test_create_swift_public(self):
options = openstack.OpenstackOptions(
user_name="user", tenant_name="tenant", project_name="project",
auth_url="url", password="password", identity_api_version="3",
endpoint_type="adminURL", insecure=False, cert='cert',
verify=True)
client = osclients.ClientManager(options, None, None, None)
client.create_swift()
def test_dry_run(self): def test_dry_run(self):
osclients.DryRunSwiftclientConnectionWrapper(mock.Mock()) osclients.DryRunSwiftclientConnectionWrapper(mock.Mock())
def test_get_cinder(self): def test_get_cinder(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.get_cinder()
client.get_cinder()
def test_get_swift(self): def test_get_swift(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.get_swift()
client.get_swift()
def get_glance(self): def get_glance(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.get_glance()
client.get_glance()
def get_nova(self): def get_nova(self):
client = osclients.ClientManager(self.fake_options, None, None, None) self.client_manager.get_nova()
client.get_nova()

View File

@ -1,7 +1,7 @@
"""Freezer restore.py related tests """Freezer restore.py related tests
(c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
(c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P
Licensed under the Apache License, Version 2.0 (the "License"); Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. you may not use this file except in compliance with the License.
You may obtain a copy of the License at You may obtain a copy of the License at

View File

@ -1,4 +1,5 @@
# (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. # (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
# (c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P
# #
# Licensed under the Apache License, Version 2.0 (the "License"); # Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License. # you may not use this file except in compliance with the License.
@ -15,7 +16,6 @@
import unittest import unittest
from freezer.openstack import openstack
from freezer.openstack import osclients from freezer.openstack import osclients
from freezer.storage import swift from freezer.storage import swift
from freezer.storage import base from freezer.storage import base
@ -24,11 +24,12 @@ from freezer.storage import base
class TestSwiftStorage(unittest.TestCase): class TestSwiftStorage(unittest.TestCase):
def setUp(self): def setUp(self):
opts = osclients.OpenstackOpts.create_from_env().get_opts_dicts()
self.storage = swift.SwiftStorage( self.storage = swift.SwiftStorage(
osclients.ClientManager( osclients.OSClientManager(opts.pop('auth_url'),
openstack.OpenstackOptions.create_from_env() opts.pop('auth_method', 'password'),
), **opts
),
"freezer_ops-aw1ops1-gerrit0001.aw1.hpcloud.net", "freezer_ops-aw1ops1-gerrit0001.aw1.hpcloud.net",
"/tmp/", "/tmp/",
100, skip_prepare=True 100, skip_prepare=True

View File

@ -1,4 +1,5 @@
# (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P. # (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
# (c) Copyright 2016 Hewlett-Packard Enterprise Development Company, L.P
# #
# Licensed under the Apache License, Version 2.0 (the "License"); # Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License. # you may not use this file except in compliance with the License.
@ -16,7 +17,7 @@
import datetime import datetime
import unittest import unittest
from freezer.openstack import openstack from freezer.openstack.osclients import OpenstackOpts
from freezer.tests.commons import * from freezer.tests.commons import *
from freezer.utils import utils from freezer.utils import utils
@ -65,14 +66,19 @@ class TestUtils(unittest.TestCase):
self.assertRaises(ValueError, utils.human2bytes, '12 foo') self.assertRaises(ValueError, utils.human2bytes, '12 foo')
def test_OpenstackOptions_creation_success(self): def test_OpenstackOptions_creation_success(self):
class FreezerOpts:
def __init__(self, opts):
self.__dict__.update(opts)
env_dict = dict(OS_USERNAME='testusername', env_dict = dict(OS_USERNAME='testusername',
OS_TENANT_NAME='testtenantename', OS_TENANT_NAME='testtenantename',
OS_AUTH_URL='testauthurl', OS_AUTH_URL='testauthurl',
OS_PASSWORD='testpassword', OS_PASSWORD='testpassword',
OS_REGION_NAME='testregion', OS_REGION_NAME='testregion',
OS_TENANT_ID='0123456789') OS_TENANT_ID='0123456789',
options = openstack.OpenstackOptions.create_from_dict(env_dict) OS_AUTH_VERSION='2.0')
assert options.user_name == env_dict['OS_USERNAME'] options = OpenstackOpts.create_from_dict(env_dict).get_opts_dicts()
options = FreezerOpts(options)
assert options.username == env_dict['OS_USERNAME']
assert options.tenant_name == env_dict['OS_TENANT_NAME'] assert options.tenant_name == env_dict['OS_TENANT_NAME']
assert options.auth_url == env_dict['OS_AUTH_URL'] assert options.auth_url == env_dict['OS_AUTH_URL']
assert options.password == env_dict['OS_PASSWORD'] assert options.password == env_dict['OS_PASSWORD']
@ -82,14 +88,14 @@ class TestUtils(unittest.TestCase):
env_dict = dict(OS_USERNAME='testusername', env_dict = dict(OS_USERNAME='testusername',
OS_TENANT_NAME='testtenantename', OS_TENANT_NAME='testtenantename',
OS_AUTH_URL='testauthurl', OS_AUTH_URL='testauthurl',
OS_PASSWORD='testpassword') OS_PASSWORD='testpassword',
options = openstack.OpenstackOptions.create_from_dict(env_dict) OS_AUTH_VERSION='2.0')
assert options.user_name == env_dict['OS_USERNAME'] options = OpenstackOpts.create_from_dict(env_dict).get_opts_dicts()
options = FreezerOpts(options)
assert options.username == env_dict['OS_USERNAME']
assert options.tenant_name == env_dict['OS_TENANT_NAME'] assert options.tenant_name == env_dict['OS_TENANT_NAME']
assert options.auth_url == env_dict['OS_AUTH_URL'] assert options.auth_url == env_dict['OS_AUTH_URL']
assert options.password == env_dict['OS_PASSWORD'] assert options.password == env_dict['OS_PASSWORD']
assert options.region_name is None
assert options.tenant_id is None
def test_date_to_timestamp(self): def test_date_to_timestamp(self):
# ensure that timestamp is check with appropriate timezone offset # ensure that timestamp is check with appropriate timezone offset
@ -136,8 +142,8 @@ class TestDateTime:
d = utils.DateTime(1425750464) d = utils.DateTime(1425750464)
assert 1425750464 == d.timestamp assert 1425750464 == d.timestamp
#ensure that time is check with appropriate timezone offset #ensure that time is check with appropriate timezone offset
t = time.strftime("%Y-%m-%d %H:%M:%S", t = time.strftime("%Y-%m-%d %H:%M:%S",
time.localtime((time.mktime(time.strptime("2015-03-07 17:47:44", time.localtime((time.mktime(time.strptime("2015-03-07 17:47:44",
"%Y-%m-%d %H:%M:%S")))-time.timezone)) "%Y-%m-%d %H:%M:%S")))-time.timezone))
assert t == '{}'.format(d) assert t == '{}'.format(d)