Improve and fix images & flavors handling

Fix images matching algo (exact match favored compared to partial match)
Cache images & flavors list
Fix OpenStack session init logic (was called too late and only on deploy)

Change-Id: I6c723f03925a8b2c4b3408fcfb6c78d5af47328d
This commit is contained in:
Mathieu Velten 2016-07-26 13:28:08 +02:00
parent 7a2fa32b83
commit e60d85ed7a
4 changed files with 73 additions and 34 deletions

View File

@ -12,17 +12,17 @@
import logging import logging
# NOTE(aloga): this should be safe. If we do not have the clients, we won't
# have the session below, therefore the clients won't be ever called.
try: try:
import novaclient.client import novaclient.client
client_available = True
except ImportError: except ImportError:
client_available = False
pass pass
log = logging.getLogger('heat-translator') log = logging.getLogger('heat-translator')
_FLAVORS = { PREDEF_FLAVORS = {
'm1.xlarge': {'mem_size': 16384, 'disk_size': 160, 'num_cpus': 8}, 'm1.xlarge': {'mem_size': 16384, 'disk_size': 160, 'num_cpus': 8},
'm1.large': {'mem_size': 8192, 'disk_size': 80, 'num_cpus': 4}, 'm1.large': {'mem_size': 8192, 'disk_size': 80, 'num_cpus': 4},
'm1.medium': {'mem_size': 4096, 'disk_size': 40, 'num_cpus': 2}, 'm1.medium': {'mem_size': 4096, 'disk_size': 40, 'num_cpus': 2},
@ -34,10 +34,16 @@ _FLAVORS = {
SESSION = None SESSION = None
FLAVORS = {}
def get_flavors(): def get_flavors():
ret = {} global FLAVORS
if SESSION is not None:
if FLAVORS:
return FLAVORS
if SESSION is not None and client_available:
try: try:
client = novaclient.client.Client("2", session=SESSION) client = novaclient.client.Client("2", session=SESSION)
except Exception as e: except Exception as e:
@ -46,9 +52,13 @@ def get_flavors():
'Openstack Exception: %s') % str(e)) 'Openstack Exception: %s') % str(e))
else: else:
for flv in client.flavors.list(detailed=True): for flv in client.flavors.list(detailed=True):
ret[str(flv.name)] = { FLAVORS[str(flv.name)] = {
"mem_size": flv.ram, "mem_size": flv.ram,
"disk_size": flv.disk, "disk_size": flv.disk,
"num_cpus": flv.vcpus "num_cpus": flv.vcpus
} }
return ret or _FLAVORS
if not FLAVORS:
FLAVORS = PREDEF_FLAVORS
return FLAVORS

View File

@ -12,17 +12,17 @@
import logging import logging
# NOTE(aloga): this should be safe. If we do not have the clients, we won't
# have the session below, therefore the clients won't be ever called.
try: try:
import glanceclient.client import glanceclient.client
client_available = True
except ImportError: except ImportError:
client_available = False
pass pass
log = logging.getLogger('heat-translator') log = logging.getLogger('heat-translator')
_IMAGES = { PREDEF_IMAGES = {
'ubuntu-software-config-os-init': {'architecture': 'x86_64', 'ubuntu-software-config-os-init': {'architecture': 'x86_64',
'type': 'Linux', 'type': 'Linux',
'distribution': 'Ubuntu', 'distribution': 'Ubuntu',
@ -59,11 +59,16 @@ _IMAGES = {
SESSION = None SESSION = None
IMAGES = {}
def get_images(): def get_images():
ret = {} global IMAGES
if SESSION is not None: if IMAGES:
return IMAGES
if SESSION is not None and client_available:
try: try:
client = glanceclient.client.Client("2", session=SESSION) client = glanceclient.client.Client("2", session=SESSION)
except Exception as e: except Exception as e:
@ -72,10 +77,15 @@ def get_images():
'Openstack Exception: %s') % str(e)) 'Openstack Exception: %s') % str(e))
else: else:
for image in client.images.list(): for image in client.images.list():
image_name = image.name.encode('ascii', 'ignore')
metadata = ["architecture", "type", "distribution", "version"] metadata = ["architecture", "type", "distribution", "version"]
if any(key in image.keys() for key in metadata): if any(key in image.keys() for key in metadata):
ret = [image["name"]] = {} IMAGES[image_name] = {}
for key in metadata: for key in metadata:
if key in image.keys(): if key in image.keys():
ret[image["name"]][key] = image[key] IMAGES[image_name][key] = image[key]
return ret or _IMAGES
if not IMAGES:
IMAGES = PREDEF_IMAGES
return IMAGES

View File

@ -186,7 +186,10 @@ class ToscaCompute(HotResource):
return this_list return this_list
matching_images = [] matching_images = []
for image in this_list: for image in this_list:
if attr in this_dict[image]:
if this_dict[image][attr].lower() == str(prop).lower(): if this_dict[image][attr].lower() == str(prop).lower():
matching_images.insert(0, image)
else:
matching_images.append(image) matching_images.append(image)
return matching_images return matching_images

View File

@ -22,12 +22,19 @@ import yaml
# NOTE(aloga): As per upstream developers requirement this needs to work # NOTE(aloga): As per upstream developers requirement this needs to work
# without the clients, therefore we need to pass if we cannot import them # without the clients, therefore we need to pass if we cannot import them
try: try:
import heatclient.client
from keystoneauth1 import loading from keystoneauth1 import loading
except ImportError: except ImportError:
has_clients = False keystone_client_avail = False
else: else:
has_clients = True keystone_client_avail = True
try:
import heatclient.client
except ImportError:
heat_client_avail = False
else:
heat_client_avail = True
from toscaparser.tosca_template import ToscaTemplate from toscaparser.tosca_template import ToscaTemplate
from toscaparser.utils.gettextutils import _ from toscaparser.utils.gettextutils import _
@ -103,7 +110,7 @@ class TranslatorShell(object):
return parser return parser
def _append_global_identity_args(self, parser, argv): def _append_global_identity_args(self, parser, argv):
if not has_clients: if not keystone_client_avail:
return return
loading.register_session_argparse_arguments(parser) loading.register_session_argparse_arguments(parser)
@ -138,13 +145,8 @@ class TranslatorShell(object):
'validation.') % {'template_file': template_file}) 'validation.') % {'template_file': template_file})
print(msg) print(msg)
else: else:
hot = self._translate(template_type, template_file, if keystone_client_avail:
parsed_params, a_file, deploy) try:
if hot and deploy:
if not has_clients:
raise RuntimeError(_('Could not find OpenStack '
'clients and libs, aborting '))
keystone_auth = ( keystone_auth = (
loading.load_auth_from_argparse_arguments(args) loading.load_auth_from_argparse_arguments(args)
) )
@ -156,6 +158,20 @@ class TranslatorShell(object):
) )
images.SESSION = keystone_session images.SESSION = keystone_session
flavors.SESSION = keystone_session flavors.SESSION = keystone_session
except Exception:
keystone_session = None
hot = self._translate(template_type, template_file,
parsed_params, a_file, deploy)
if hot and deploy:
if not keystone_client_avail or not heat_client_avail:
raise RuntimeError(_('Could not find Heat or Keystone'
'client to deploy, aborting '))
if not keystone_session:
raise RuntimeError(_('Impossible to login with '
'Keystone to deploy on Heat, '
'please check your credentials'))
self.deploy_on_heat(keystone_session, keystone_auth, self.deploy_on_heat(keystone_session, keystone_auth,
hot, parsed_params) hot, parsed_params)