merge with nova trunk
This commit is contained in:
		
							
								
								
									
										3
									
								
								Authors
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								Authors
									
									
									
									
									
								
							| @@ -31,6 +31,7 @@ Hisaharu Ishii <ishii.hisaharu@lab.ntt.co.jp> | ||||
| Hisaki Ohara <hisaki.ohara@intel.com> | ||||
| Ilya Alekseyev <ialekseev@griddynamics.com> | ||||
| Isaku Yamahata <yamahata@valinux.co.jp> | ||||
| Jason Cannavale <jason.cannavale@rackspace.com> | ||||
| Jason Koelker <jason@koelker.net> | ||||
| Jay Pipes <jaypipes@gmail.com> | ||||
| Jesse Andrews <anotherjesse@gmail.com> | ||||
| @@ -59,6 +60,7 @@ Mark Washenberger <mark.washenberger@rackspace.com> | ||||
| Masanori Itoh <itoumsn@nttdata.co.jp> | ||||
| Matt Dietz <matt.dietz@rackspace.com> | ||||
| Michael Gundlach <michael.gundlach@rackspace.com> | ||||
| Mike Scherbakov <mihgen@gmail.com> | ||||
| Monsyne Dragon <mdragon@rackspace.com> | ||||
| Monty Taylor <mordred@inaugust.com> | ||||
| MORITA Kazutaka <morita.kazutaka@gmail.com> | ||||
| @@ -84,6 +86,7 @@ Trey Morris <trey.morris@rackspace.com> | ||||
| Tushar Patil <tushar.vitthal.patil@gmail.com> | ||||
| Vasiliy Shlykov <vash@vasiliyshlykov.org> | ||||
| Vishvananda Ishaya <vishvananda@gmail.com> | ||||
| Vivek Y S <vivek.ys@gmail.com> | ||||
| William Wolf <throughnothing@gmail.com> | ||||
| Yoshiaki Tamura <yoshi@midokura.jp> | ||||
| Youcef Laribi <Youcef.Laribi@eu.citrix.com> | ||||
|   | ||||
| @@ -53,7 +53,6 @@ | ||||
|   CLI interface for nova management. | ||||
| """ | ||||
|  | ||||
| import datetime | ||||
| import gettext | ||||
| import glob | ||||
| import json | ||||
| @@ -78,6 +77,7 @@ from nova import crypto | ||||
| from nova import db | ||||
| from nova import exception | ||||
| from nova import flags | ||||
| from nova import image | ||||
| from nova import log as logging | ||||
| from nova import quota | ||||
| from nova import rpc | ||||
| @@ -96,6 +96,7 @@ flags.DECLARE('network_size', 'nova.network.manager') | ||||
| flags.DECLARE('vlan_start', 'nova.network.manager') | ||||
| flags.DECLARE('vpn_start', 'nova.network.manager') | ||||
| flags.DECLARE('fixed_range_v6', 'nova.network.manager') | ||||
| flags.DECLARE('gateway_v6', 'nova.network.manager') | ||||
| flags.DECLARE('images_path', 'nova.image.local') | ||||
| flags.DECLARE('libvirt_type', 'nova.virt.libvirt.connection') | ||||
| flags.DEFINE_flag(flags.HelpFlag()) | ||||
| @@ -536,7 +537,7 @@ class FloatingIpCommands(object): | ||||
|         for floating_ip in floating_ips: | ||||
|             instance = None | ||||
|             if floating_ip['fixed_ip']: | ||||
|                 instance = floating_ip['fixed_ip']['instance']['ec2_id'] | ||||
|                 instance = floating_ip['fixed_ip']['instance']['hostname'] | ||||
|             print "%s\t%s\t%s" % (floating_ip['host'], | ||||
|                                   floating_ip['address'], | ||||
|                                   instance) | ||||
| @@ -545,13 +546,10 @@ class FloatingIpCommands(object): | ||||
| class NetworkCommands(object): | ||||
|     """Class for managing networks.""" | ||||
|  | ||||
|     def create(self, fixed_range=None, num_networks=None, | ||||
|                network_size=None, vlan_start=None, | ||||
|                vpn_start=None, fixed_range_v6=None, label='public'): | ||||
|         """Creates fixed ips for host by range | ||||
|         arguments: fixed_range=FLAG, [num_networks=FLAG], | ||||
|                    [network_size=FLAG], [vlan_start=FLAG], | ||||
|                    [vpn_start=FLAG], [fixed_range_v6=FLAG]""" | ||||
|     def create(self, fixed_range=None, num_networks=None, network_size=None, | ||||
|             vlan_start=None, vpn_start=None, fixed_range_v6=None, | ||||
|             gateway_v6=None, label='public'): | ||||
|         """Creates fixed ips for host by range""" | ||||
|         if not fixed_range: | ||||
|             msg = _('Fixed range in the form of 10.0.0.0/8 is ' | ||||
|                     'required to create networks.') | ||||
| @@ -567,6 +565,8 @@ class NetworkCommands(object): | ||||
|             vpn_start = FLAGS.vpn_start | ||||
|         if not fixed_range_v6: | ||||
|             fixed_range_v6 = FLAGS.fixed_range_v6 | ||||
|         if not gateway_v6: | ||||
|             gateway_v6 = FLAGS.gateway_v6 | ||||
|         net_manager = utils.import_object(FLAGS.network_manager) | ||||
|         try: | ||||
|             net_manager.create_networks(context.get_admin_context(), | ||||
| @@ -576,6 +576,7 @@ class NetworkCommands(object): | ||||
|                                         vlan_start=int(vlan_start), | ||||
|                                         vpn_start=int(vpn_start), | ||||
|                                         cidr_v6=fixed_range_v6, | ||||
|                                         gateway_v6=gateway_v6, | ||||
|                                         label=label) | ||||
|         except ValueError, e: | ||||
|             print e | ||||
| @@ -689,7 +690,7 @@ class ServiceCommands(object): | ||||
|         """Show a list of all running services. Filter by host & service name. | ||||
|         args: [host] [service]""" | ||||
|         ctxt = context.get_admin_context() | ||||
|         now = datetime.datetime.utcnow() | ||||
|         now = utils.utcnow() | ||||
|         services = db.service_get_all(ctxt) | ||||
|         if host: | ||||
|             services = [s for s in services if s['host'] == host] | ||||
| @@ -936,7 +937,7 @@ class ImageCommands(object): | ||||
|     """Methods for dealing with a cloud in an odd state""" | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self.image_service = utils.import_object(FLAGS.image_service) | ||||
|         self.image_service = image.get_default_image_service() | ||||
|  | ||||
|     def _register(self, container_format, disk_format, | ||||
|                   path, owner, name=None, is_public='T', | ||||
| @@ -1081,24 +1082,35 @@ class ImageCommands(object): | ||||
|         self._convert_images(machine_images) | ||||
|  | ||||
|  | ||||
| class ConfigCommands(object): | ||||
|     """Class for exposing the flags defined by flag_file(s).""" | ||||
|  | ||||
|     def __init__(self): | ||||
|         pass | ||||
|  | ||||
|     def list(self): | ||||
|         print FLAGS.FlagsIntoString() | ||||
|  | ||||
|  | ||||
| CATEGORIES = [ | ||||
|     ('user', UserCommands), | ||||
|     ('account', AccountCommands), | ||||
|     ('project', ProjectCommands), | ||||
|     ('role', RoleCommands), | ||||
|     ('shell', ShellCommands), | ||||
|     ('vpn', VpnCommands), | ||||
|     ('fixed', FixedIpCommands), | ||||
|     ('floating', FloatingIpCommands), | ||||
|     ('network', NetworkCommands), | ||||
|     ('vm', VmCommands), | ||||
|     ('service', ServiceCommands), | ||||
|     ('config', ConfigCommands), | ||||
|     ('db', DbCommands), | ||||
|     ('volume', VolumeCommands), | ||||
|     ('fixed', FixedIpCommands), | ||||
|     ('flavor', InstanceTypeCommands), | ||||
|     ('floating', FloatingIpCommands), | ||||
|     ('instance_type', InstanceTypeCommands), | ||||
|     ('image', ImageCommands), | ||||
|     ('flavor', InstanceTypeCommands), | ||||
|     ('version', VersionCommands)] | ||||
|     ('network', NetworkCommands), | ||||
|     ('project', ProjectCommands), | ||||
|     ('role', RoleCommands), | ||||
|     ('service', ServiceCommands), | ||||
|     ('shell', ShellCommands), | ||||
|     ('user', UserCommands), | ||||
|     ('version', VersionCommands), | ||||
|     ('vm', VmCommands), | ||||
|     ('volume', VolumeCommands), | ||||
|     ('vpn', VpnCommands)] | ||||
|  | ||||
|  | ||||
| def lazy_match(name, key_value_tuples): | ||||
|   | ||||
| @@ -24,6 +24,7 @@ other backends by creating another class that exposes the same | ||||
| public methods. | ||||
| """ | ||||
|  | ||||
| import functools | ||||
| import sys | ||||
|  | ||||
| from nova import exception | ||||
| @@ -68,6 +69,12 @@ flags.DEFINE_string('ldap_developer', | ||||
| LOG = logging.getLogger("nova.ldapdriver") | ||||
|  | ||||
|  | ||||
| if FLAGS.memcached_servers: | ||||
|     import memcache | ||||
| else: | ||||
|     from nova import fakememcache as memcache | ||||
|  | ||||
|  | ||||
| # TODO(vish): make an abstract base class with the same public methods | ||||
| #             to define a set interface for AuthDrivers. I'm delaying | ||||
| #             creating this now because I'm expecting an auth refactor | ||||
| @@ -85,6 +92,7 @@ def _clean(attr): | ||||
|  | ||||
| def sanitize(fn): | ||||
|     """Decorator to sanitize all args""" | ||||
|     @functools.wraps(fn) | ||||
|     def _wrapped(self, *args, **kwargs): | ||||
|         args = [_clean(x) for x in args] | ||||
|         kwargs = dict((k, _clean(v)) for (k, v) in kwargs) | ||||
| @@ -103,29 +111,56 @@ class LdapDriver(object): | ||||
|     isadmin_attribute = 'isNovaAdmin' | ||||
|     project_attribute = 'owner' | ||||
|     project_objectclass = 'groupOfNames' | ||||
|     conn = None | ||||
|     mc = None | ||||
|  | ||||
|     def __init__(self): | ||||
|         """Imports the LDAP module""" | ||||
|         self.ldap = __import__('ldap') | ||||
|         self.conn = None | ||||
|         if FLAGS.ldap_schema_version == 1: | ||||
|             LdapDriver.project_pattern = '(objectclass=novaProject)' | ||||
|             LdapDriver.isadmin_attribute = 'isAdmin' | ||||
|             LdapDriver.project_attribute = 'projectManager' | ||||
|             LdapDriver.project_objectclass = 'novaProject' | ||||
|         self.__cache = None | ||||
|         if LdapDriver.conn is None: | ||||
|             LdapDriver.conn = self.ldap.initialize(FLAGS.ldap_url) | ||||
|             LdapDriver.conn.simple_bind_s(FLAGS.ldap_user_dn, | ||||
|                                           FLAGS.ldap_password) | ||||
|         if LdapDriver.mc is None: | ||||
|             LdapDriver.mc = memcache.Client(FLAGS.memcached_servers, debug=0) | ||||
|  | ||||
|     def __enter__(self): | ||||
|         """Creates the connection to LDAP""" | ||||
|         self.conn = self.ldap.initialize(FLAGS.ldap_url) | ||||
|         self.conn.simple_bind_s(FLAGS.ldap_user_dn, FLAGS.ldap_password) | ||||
|         # TODO(yorik-sar): Should be per-request cache, not per-driver-request | ||||
|         self.__cache = {} | ||||
|         return self | ||||
|  | ||||
|     def __exit__(self, exc_type, exc_value, traceback): | ||||
|         """Destroys the connection to LDAP""" | ||||
|         self.conn.unbind_s() | ||||
|         self.__cache = None | ||||
|         return False | ||||
|  | ||||
|     def __local_cache(key_fmt):  # pylint: disable=E0213 | ||||
|         """Wrap function to cache it's result in self.__cache. | ||||
|         Works only with functions with one fixed argument. | ||||
|         """ | ||||
|         def do_wrap(fn): | ||||
|             @functools.wraps(fn) | ||||
|             def inner(self, arg, **kwargs): | ||||
|                 cache_key = key_fmt % (arg,) | ||||
|                 try: | ||||
|                     res = self.__cache[cache_key] | ||||
|                     LOG.debug('Local cache hit for %s by key %s' % | ||||
|                               (fn.__name__, cache_key)) | ||||
|                     return res | ||||
|                 except KeyError: | ||||
|                     res = fn(self, arg, **kwargs) | ||||
|                     self.__cache[cache_key] = res | ||||
|                     return res | ||||
|             return inner | ||||
|         return do_wrap | ||||
|  | ||||
|     @sanitize | ||||
|     @__local_cache('uid_user-%s') | ||||
|     def get_user(self, uid): | ||||
|         """Retrieve user by id""" | ||||
|         attr = self.__get_ldap_user(uid) | ||||
| @@ -134,15 +169,31 @@ class LdapDriver(object): | ||||
|     @sanitize | ||||
|     def get_user_from_access_key(self, access): | ||||
|         """Retrieve user by access key""" | ||||
|         cache_key = 'uak_dn_%s' % (access,) | ||||
|         user_dn = self.mc.get(cache_key) | ||||
|         if user_dn: | ||||
|             user = self.__to_user( | ||||
|                 self.__find_object(user_dn, scope=self.ldap.SCOPE_BASE)) | ||||
|             if user: | ||||
|                 if user['access'] == access: | ||||
|                     return user | ||||
|                 else: | ||||
|                     self.mc.set(cache_key, None) | ||||
|         query = '(accessKey=%s)' % access | ||||
|         dn = FLAGS.ldap_user_subtree | ||||
|         return self.__to_user(self.__find_object(dn, query)) | ||||
|         user_obj = self.__find_object(dn, query) | ||||
|         user = self.__to_user(user_obj) | ||||
|         if user: | ||||
|             self.mc.set(cache_key, user_obj['dn'][0]) | ||||
|         return user | ||||
|  | ||||
|     @sanitize | ||||
|     @__local_cache('pid_project-%s') | ||||
|     def get_project(self, pid): | ||||
|         """Retrieve project by id""" | ||||
|         dn = self.__project_to_dn(pid) | ||||
|         attr = self.__find_object(dn, LdapDriver.project_pattern) | ||||
|         dn = self.__project_to_dn(pid, search=False) | ||||
|         attr = self.__find_object(dn, LdapDriver.project_pattern, | ||||
|                                   scope=self.ldap.SCOPE_BASE) | ||||
|         return self.__to_project(attr) | ||||
|  | ||||
|     @sanitize | ||||
| @@ -395,6 +446,7 @@ class LdapDriver(object): | ||||
|         """Check if project exists""" | ||||
|         return self.get_project(project_id) is not None | ||||
|  | ||||
|     @__local_cache('uid_attrs-%s') | ||||
|     def __get_ldap_user(self, uid): | ||||
|         """Retrieve LDAP user entry by id""" | ||||
|         dn = FLAGS.ldap_user_subtree | ||||
| @@ -426,12 +478,20 @@ class LdapDriver(object): | ||||
|         if scope is None: | ||||
|             # One of the flags is 0! | ||||
|             scope = self.ldap.SCOPE_SUBTREE | ||||
|         if query is None: | ||||
|             query = "(objectClass=*)" | ||||
|         try: | ||||
|             res = self.conn.search_s(dn, scope, query) | ||||
|         except self.ldap.NO_SUCH_OBJECT: | ||||
|             return [] | ||||
|         # Just return the attributes | ||||
|         return [attributes for dn, attributes in res] | ||||
|         # FIXME(yorik-sar): Whole driver should be refactored to | ||||
|         #                   prevent this hack | ||||
|         res1 = [] | ||||
|         for dn, attrs in res: | ||||
|             attrs['dn'] = [dn] | ||||
|             res1.append(attrs) | ||||
|         return res1 | ||||
|  | ||||
|     def __find_role_dns(self, tree): | ||||
|         """Find dns of role objects in given tree""" | ||||
| @@ -564,6 +624,7 @@ class LdapDriver(object): | ||||
|             'description': attr.get('description', [None])[0], | ||||
|             'member_ids': [self.__dn_to_uid(x) for x in member_dns]} | ||||
|  | ||||
|     @__local_cache('uid_dn-%s') | ||||
|     def __uid_to_dn(self, uid, search=True): | ||||
|         """Convert uid to dn""" | ||||
|         # By default return a generated DN | ||||
| @@ -576,6 +637,7 @@ class LdapDriver(object): | ||||
|                 userdn = user[0] | ||||
|         return userdn | ||||
|  | ||||
|     @__local_cache('pid_dn-%s') | ||||
|     def __project_to_dn(self, pid, search=True): | ||||
|         """Convert pid to dn""" | ||||
|         # By default return a generated DN | ||||
| @@ -603,16 +665,18 @@ class LdapDriver(object): | ||||
|         else: | ||||
|             return None | ||||
|  | ||||
|     @__local_cache('dn_uid-%s') | ||||
|     def __dn_to_uid(self, dn): | ||||
|         """Convert user dn to uid""" | ||||
|         query = '(objectclass=novaUser)' | ||||
|         user = self.__find_object(dn, query) | ||||
|         user = self.__find_object(dn, query, scope=self.ldap.SCOPE_BASE) | ||||
|         return user[FLAGS.ldap_user_id_attribute][0] | ||||
|  | ||||
|  | ||||
| class FakeLdapDriver(LdapDriver): | ||||
|     """Fake Ldap Auth driver""" | ||||
|  | ||||
|     def __init__(self):  # pylint: disable=W0231 | ||||
|         __import__('nova.auth.fakeldap') | ||||
|         self.ldap = sys.modules['nova.auth.fakeldap'] | ||||
|     def __init__(self): | ||||
|         import nova.auth.fakeldap | ||||
|         sys.modules['ldap'] = nova.auth.fakeldap | ||||
|         super(FakeLdapDriver, self).__init__() | ||||
|   | ||||
| @@ -73,6 +73,12 @@ flags.DEFINE_string('auth_driver', 'nova.auth.dbdriver.DbDriver', | ||||
| LOG = logging.getLogger('nova.auth.manager') | ||||
|  | ||||
|  | ||||
| if FLAGS.memcached_servers: | ||||
|     import memcache | ||||
| else: | ||||
|     from nova import fakememcache as memcache | ||||
|  | ||||
|  | ||||
| class AuthBase(object): | ||||
|     """Base class for objects relating to auth | ||||
|  | ||||
| @@ -206,6 +212,7 @@ class AuthManager(object): | ||||
|     """ | ||||
|  | ||||
|     _instance = None | ||||
|     mc = None | ||||
|  | ||||
|     def __new__(cls, *args, **kwargs): | ||||
|         """Returns the AuthManager singleton""" | ||||
| @@ -222,13 +229,8 @@ class AuthManager(object): | ||||
|         self.network_manager = utils.import_object(FLAGS.network_manager) | ||||
|         if driver or not getattr(self, 'driver', None): | ||||
|             self.driver = utils.import_class(driver or FLAGS.auth_driver) | ||||
|  | ||||
|         if FLAGS.memcached_servers: | ||||
|             import memcache | ||||
|         else: | ||||
|             from nova import fakememcache as memcache | ||||
|         self.mc = memcache.Client(FLAGS.memcached_servers, | ||||
|                                   debug=0) | ||||
|         if AuthManager.mc is None: | ||||
|             AuthManager.mc = memcache.Client(FLAGS.memcached_servers, debug=0) | ||||
|  | ||||
|     def authenticate(self, access, signature, params, verb='GET', | ||||
|                      server_string='127.0.0.1:8773', path='/', | ||||
|   | ||||
| @@ -1,4 +1,6 @@ | ||||
| NOVA_KEY_DIR=$(pushd $(dirname $BASH_SOURCE)>/dev/null; pwd; popd>/dev/null) | ||||
| NOVARC=$(readlink -f "${BASH_SOURCE:-${0}}" 2>/dev/null) || | ||||
|     NOVARC=$(python -c 'import os,sys; print os.path.abspath(os.path.realpath(sys.argv[1]))' "${BASH_SOURCE:-${0}}") | ||||
| NOVA_KEY_DIR=${NOVARC%%/*} | ||||
| export EC2_ACCESS_KEY="%(access)s:%(project)s" | ||||
| export EC2_SECRET_KEY="%(secret)s" | ||||
| export EC2_URL="%(ec2)s" | ||||
| @@ -12,4 +14,5 @@ alias ec2-bundle-image="ec2-bundle-image --cert ${EC2_CERT} --privatekey ${EC2_P | ||||
| alias ec2-upload-bundle="ec2-upload-bundle -a ${EC2_ACCESS_KEY} -s ${EC2_SECRET_KEY} --url ${S3_URL} --ec2cert ${NOVA_CERT}" | ||||
| export NOVA_API_KEY="%(access)s" | ||||
| export NOVA_USERNAME="%(user)s" | ||||
| export NOVA_PROJECT_ID="%(project)s" | ||||
| export NOVA_URL="%(os)s" | ||||
|   | ||||
| @@ -270,8 +270,10 @@ DEFINE_list('region_list', | ||||
| DEFINE_string('connection_type', 'libvirt', 'libvirt, xenapi or fake') | ||||
| DEFINE_string('aws_access_key_id', 'admin', 'AWS Access ID') | ||||
| DEFINE_string('aws_secret_access_key', 'admin', 'AWS Access Key') | ||||
| DEFINE_integer('glance_port', 9292, 'glance port') | ||||
| DEFINE_string('glance_host', '$my_ip', 'glance host') | ||||
| # NOTE(sirp): my_ip interpolation doesn't work within nested structures | ||||
| DEFINE_list('glance_api_servers', | ||||
|             ['127.0.0.1:9292'], | ||||
|             'list of glance api servers available to nova (host:port)') | ||||
| DEFINE_integer('s3_port', 3333, 's3 port') | ||||
| DEFINE_string('s3_host', '$my_ip', 's3 host (for infrastructure)') | ||||
| DEFINE_string('s3_dmz', '$my_ip', 's3 dmz ip (for instances)') | ||||
| @@ -296,6 +298,7 @@ DEFINE_bool('fake_network', False, | ||||
|             'should we use fake network devices and addresses') | ||||
| DEFINE_string('rabbit_host', 'localhost', 'rabbit host') | ||||
| DEFINE_integer('rabbit_port', 5672, 'rabbit port') | ||||
| DEFINE_bool('rabbit_use_ssl', False, 'connect over SSL') | ||||
| DEFINE_string('rabbit_userid', 'guest', 'rabbit userid') | ||||
| DEFINE_string('rabbit_password', 'guest', 'rabbit password') | ||||
| DEFINE_string('rabbit_virtual_host', '/', 'rabbit virtual host') | ||||
| @@ -380,3 +383,5 @@ DEFINE_string('zone_name', 'nova', 'name of this zone') | ||||
| DEFINE_list('zone_capabilities', | ||||
|                 ['hypervisor=xenserver;kvm', 'os=linux;windows'], | ||||
|                  'Key/Multi-value list representng capabilities of this zone') | ||||
| DEFINE_string('build_plan_encryption_key', None, | ||||
|         '128bit (hex) encryption key for scheduler build plans.') | ||||
|   | ||||
							
								
								
									
										10
									
								
								nova/log.py
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								nova/log.py
									
									
									
									
									
								
							| @@ -35,6 +35,7 @@ import os | ||||
| import sys | ||||
| import traceback | ||||
|  | ||||
| import nova | ||||
| from nova import flags | ||||
| from nova import version | ||||
|  | ||||
| @@ -63,6 +64,7 @@ flags.DEFINE_list('default_log_levels', | ||||
|                    'eventlet.wsgi.server=WARN'], | ||||
|                   'list of logger=LEVEL pairs') | ||||
| flags.DEFINE_bool('use_syslog', False, 'output to syslog') | ||||
| flags.DEFINE_bool('publish_errors', False, 'publish error events') | ||||
| flags.DEFINE_string('logfile', None, 'output to named file') | ||||
|  | ||||
|  | ||||
| @@ -258,12 +260,20 @@ class NovaRootLogger(NovaLogger): | ||||
|         else: | ||||
|             self.removeHandler(self.filelog) | ||||
|             self.addHandler(self.streamlog) | ||||
|         if FLAGS.publish_errors: | ||||
|             self.addHandler(PublishErrorsHandler(ERROR)) | ||||
|         if FLAGS.verbose: | ||||
|             self.setLevel(DEBUG) | ||||
|         else: | ||||
|             self.setLevel(INFO) | ||||
|  | ||||
|  | ||||
| class PublishErrorsHandler(logging.Handler): | ||||
|     def emit(self, record): | ||||
|         nova.notifier.api.notify('nova.error.publisher', 'error_notification', | ||||
|             nova.notifier.api.ERROR, dict(error=record.msg)) | ||||
|  | ||||
|  | ||||
| def handle_exception(type, value, tb): | ||||
|     extra = {} | ||||
|     if FLAGS.verbose: | ||||
|   | ||||
| @@ -11,9 +11,8 @@ | ||||
| #    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 datetime | ||||
| #    under the License. | ||||
|  | ||||
| import datetime | ||||
| import uuid | ||||
|  | ||||
| from nova import flags | ||||
| @@ -64,7 +63,7 @@ def notify(publisher_id, event_type, priority, payload): | ||||
|  | ||||
|     {'message_id': str(uuid.uuid4()), | ||||
|      'publisher_id': 'compute.host1', | ||||
|      'timestamp': datetime.datetime.utcnow(), | ||||
|      'timestamp': utils.utcnow(), | ||||
|      'priority': 'WARN', | ||||
|      'event_type': 'compute.create_instance', | ||||
|      'payload': {'instance_id': 12, ... }} | ||||
| @@ -79,5 +78,5 @@ def notify(publisher_id, event_type, priority, payload): | ||||
|                    event_type=event_type, | ||||
|                    priority=priority, | ||||
|                    payload=payload, | ||||
|                    timestamp=str(datetime.datetime.utcnow())) | ||||
|                    timestamp=str(utils.utcnow())) | ||||
|     driver.notify(msg) | ||||
|   | ||||
| @@ -65,6 +65,7 @@ class Connection(carrot_connection.BrokerConnection): | ||||
|         if new or not hasattr(cls, '_instance'): | ||||
|             params = dict(hostname=FLAGS.rabbit_host, | ||||
|                           port=FLAGS.rabbit_port, | ||||
|                           ssl=FLAGS.rabbit_use_ssl, | ||||
|                           userid=FLAGS.rabbit_userid, | ||||
|                           password=FLAGS.rabbit_password, | ||||
|                           virtual_host=FLAGS.rabbit_virtual_host) | ||||
|   | ||||
| @@ -84,7 +84,7 @@ def get_zone_capabilities(context): | ||||
| def select(context, specs=None): | ||||
|     """Returns a list of hosts.""" | ||||
|     return _call_scheduler('select', context=context, | ||||
|             params={"specs": specs}) | ||||
|             params={"request_spec": specs}) | ||||
|  | ||||
|  | ||||
| def update_service_capabilities(context, service_name, host, capabilities): | ||||
| @@ -111,7 +111,8 @@ def _process(func, zone): | ||||
|     return func(nova, zone) | ||||
|  | ||||
|  | ||||
| def call_zone_method(context, method, errors_to_ignore=None, *args, **kwargs): | ||||
| def call_zone_method(context, method_name, errors_to_ignore=None, | ||||
|                      novaclient_collection_name='zones', *args, **kwargs): | ||||
|     """Returns a list of (zone, call_result) objects.""" | ||||
|     if not isinstance(errors_to_ignore, (list, tuple)): | ||||
|         # This will also handle the default None | ||||
| @@ -131,18 +132,16 @@ def call_zone_method(context, method, errors_to_ignore=None, *args, **kwargs): | ||||
|             #TODO (dabo) - add logic for failure counts per zone, | ||||
|             # with escalation after a given number of failures. | ||||
|             continue | ||||
|         zone_method = getattr(nova.zones, method) | ||||
|         novaclient_collection = getattr(nova, novaclient_collection_name) | ||||
|         collection_method = getattr(novaclient_collection, method_name) | ||||
|  | ||||
|         def _error_trap(*args, **kwargs): | ||||
|             try: | ||||
|                 return zone_method(*args, **kwargs) | ||||
|                 return collection_method(*args, **kwargs) | ||||
|             except Exception as e: | ||||
|                 if type(e) in errors_to_ignore: | ||||
|                     return None | ||||
|                 # TODO (dabo) - want to be able to re-raise here. | ||||
|                 # Returning a string now; raising was causing issues. | ||||
|                 # raise e | ||||
|                 return "ERROR", "%s" % e | ||||
|                 raise | ||||
|  | ||||
|         res = pool.spawn(_error_trap, *args, **kwargs) | ||||
|         results.append((zone, res)) | ||||
|   | ||||
| @@ -14,8 +14,8 @@ | ||||
| #    under the License. | ||||
|  | ||||
| """ | ||||
| Host Filter is a driver mechanism for requesting instance resources. | ||||
| Three drivers are included: AllHosts, Flavor & JSON. AllHosts just | ||||
| Host Filter is a mechanism for requesting instance resources. | ||||
| Three filters are included: AllHosts, Flavor & JSON. AllHosts just | ||||
| returns the full, unfiltered list of hosts. Flavor is a hard coded | ||||
| matching mechanism based on flavor criteria and JSON is an ad-hoc | ||||
| filter grammar. | ||||
| @@ -41,18 +41,20 @@ import json | ||||
| from nova import exception | ||||
| from nova import flags | ||||
| from nova import log as logging | ||||
| from nova.scheduler import zone_aware_scheduler | ||||
| from nova import utils | ||||
| from nova.scheduler import zone_aware_scheduler | ||||
|  | ||||
| LOG = logging.getLogger('nova.scheduler.host_filter') | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
| flags.DEFINE_string('default_host_filter_driver', | ||||
| flags.DEFINE_string('default_host_filter', | ||||
|                     'nova.scheduler.host_filter.AllHostsFilter', | ||||
|                     'Which driver to use for filtering hosts.') | ||||
|                     'Which filter to use for filtering hosts.') | ||||
|  | ||||
|  | ||||
| class HostFilter(object): | ||||
|     """Base class for host filter drivers.""" | ||||
|     """Base class for host filters.""" | ||||
|  | ||||
|     def instance_type_to_filter(self, instance_type): | ||||
|         """Convert instance_type into a filter for most common use-case.""" | ||||
| @@ -63,14 +65,15 @@ class HostFilter(object): | ||||
|         raise NotImplementedError() | ||||
|  | ||||
|     def _full_name(self): | ||||
|         """module.classname of the filter driver""" | ||||
|         """module.classname of the filter.""" | ||||
|         return "%s.%s" % (self.__module__, self.__class__.__name__) | ||||
|  | ||||
|  | ||||
| class AllHostsFilter(HostFilter): | ||||
|     """NOP host filter driver. Returns all hosts in ZoneManager. | ||||
|     """ NOP host filter. Returns all hosts in ZoneManager. | ||||
|     This essentially does what the old Scheduler+Chance used | ||||
|     to give us.""" | ||||
|     to give us. | ||||
|     """ | ||||
|  | ||||
|     def instance_type_to_filter(self, instance_type): | ||||
|         """Return anything to prevent base-class from raising | ||||
| @@ -83,8 +86,8 @@ class AllHostsFilter(HostFilter): | ||||
|                for host, services in zone_manager.service_states.iteritems()] | ||||
|  | ||||
|  | ||||
| class FlavorFilter(HostFilter): | ||||
|     """HostFilter driver hard-coded to work with flavors.""" | ||||
| class InstanceTypeFilter(HostFilter): | ||||
|     """HostFilter hard-coded to work with InstanceType records.""" | ||||
|  | ||||
|     def instance_type_to_filter(self, instance_type): | ||||
|         """Use instance_type to filter hosts.""" | ||||
| @@ -98,9 +101,10 @@ class FlavorFilter(HostFilter): | ||||
|             capabilities = services.get('compute', {}) | ||||
|             host_ram_mb = capabilities['host_memory_free'] | ||||
|             disk_bytes = capabilities['disk_available'] | ||||
|             if host_ram_mb >= instance_type['memory_mb'] and \ | ||||
|                 disk_bytes >= instance_type['local_gb']: | ||||
|                     selected_hosts.append((host, capabilities)) | ||||
|             spec_ram = instance_type['memory_mb'] | ||||
|             spec_disk = instance_type['local_gb'] | ||||
|             if host_ram_mb >= spec_ram and disk_bytes >= spec_disk: | ||||
|                 selected_hosts.append((host, capabilities)) | ||||
|         return selected_hosts | ||||
|  | ||||
| #host entries (currently) are like: | ||||
| @@ -109,15 +113,15 @@ class FlavorFilter(HostFilter): | ||||
| #    'host_memory_total': 8244539392, | ||||
| #    'host_memory_overhead': 184225792, | ||||
| #    'host_memory_free': 3868327936, | ||||
| #    'host_memory_free_computed': 3840843776}, | ||||
| #    'host_other-config': {}, | ||||
| #    'host_memory_free_computed': 3840843776, | ||||
| #    'host_other_config': {}, | ||||
| #    'host_ip_address': '192.168.1.109', | ||||
| #    'host_cpu_info': {}, | ||||
| #    'disk_available': 32954957824, | ||||
| #    'disk_total': 50394562560, | ||||
| #    'disk_used': 17439604736}, | ||||
| #    'disk_used': 17439604736, | ||||
| #    'host_uuid': 'cedb9b39-9388-41df-8891-c5c9a0c0fe5f', | ||||
| #    'host_name-label': 'xs-mini'} | ||||
| #    'host_name_label': 'xs-mini'} | ||||
|  | ||||
| # instance_type table has: | ||||
| #name = Column(String(255), unique=True) | ||||
| @@ -131,8 +135,9 @@ class FlavorFilter(HostFilter): | ||||
|  | ||||
|  | ||||
| class JsonFilter(HostFilter): | ||||
|     """Host Filter driver to allow simple JSON-based grammar for | ||||
|        selecting hosts.""" | ||||
|     """Host Filter to allow simple JSON-based grammar for | ||||
|     selecting hosts. | ||||
|     """ | ||||
|  | ||||
|     def _equals(self, args): | ||||
|         """First term is == all the other terms.""" | ||||
| @@ -222,13 +227,14 @@ class JsonFilter(HostFilter): | ||||
|         required_disk = instance_type['local_gb'] | ||||
|         query = ['and', | ||||
|                     ['>=', '$compute.host_memory_free', required_ram], | ||||
|                     ['>=', '$compute.disk_available', required_disk] | ||||
|                     ['>=', '$compute.disk_available', required_disk], | ||||
|                 ] | ||||
|         return (self._full_name(), json.dumps(query)) | ||||
|  | ||||
|     def _parse_string(self, string, host, services): | ||||
|         """Strings prefixed with $ are capability lookups in the | ||||
|         form '$service.capability[.subcap*]'""" | ||||
|         form '$service.capability[.subcap*]' | ||||
|         """ | ||||
|         if not string: | ||||
|             return None | ||||
|         if string[0] != '$': | ||||
| @@ -271,18 +277,48 @@ class JsonFilter(HostFilter): | ||||
|         return hosts | ||||
|  | ||||
|  | ||||
| DRIVERS = [AllHostsFilter, FlavorFilter, JsonFilter] | ||||
| FILTERS = [AllHostsFilter, InstanceTypeFilter, JsonFilter] | ||||
|  | ||||
|  | ||||
| def choose_driver(driver_name=None): | ||||
|     """Since the caller may specify which driver to use we need | ||||
|        to have an authoritative list of what is permissible. This | ||||
|        function checks the driver name against a predefined set | ||||
|        of acceptable drivers.""" | ||||
| def choose_host_filter(filter_name=None): | ||||
|     """Since the caller may specify which filter to use we need | ||||
|     to have an authoritative list of what is permissible. This | ||||
|     function checks the filter name against a predefined set | ||||
|     of acceptable filters. | ||||
|     """ | ||||
|  | ||||
|     if not driver_name: | ||||
|         driver_name = FLAGS.default_host_filter_driver | ||||
|     for driver in DRIVERS: | ||||
|         if "%s.%s" % (driver.__module__, driver.__name__) == driver_name: | ||||
|             return driver() | ||||
|     raise exception.SchedulerHostFilterDriverNotFound(driver_name=driver_name) | ||||
|     if not filter_name: | ||||
|         filter_name = FLAGS.default_host_filter | ||||
|     for filter_class in FILTERS: | ||||
|         host_match = "%s.%s" % (filter_class.__module__, filter_class.__name__) | ||||
|         if host_match == filter_name: | ||||
|             return filter_class() | ||||
|     raise exception.SchedulerHostFilterNotFound(filter_name=filter_name) | ||||
|  | ||||
|  | ||||
| class HostFilterScheduler(zone_aware_scheduler.ZoneAwareScheduler): | ||||
|     """The HostFilterScheduler uses the HostFilter to filter | ||||
|     hosts for weighing. The particular filter used may be passed in | ||||
|     as an argument or the default will be used. | ||||
|  | ||||
|     request_spec = {'filter': <Filter name>, | ||||
|                     'instance_type': <InstanceType dict>} | ||||
|     """ | ||||
|  | ||||
|     def filter_hosts(self, num, request_spec): | ||||
|         """Filter the full host list (from the ZoneManager)""" | ||||
|         filter_name = request_spec.get('filter', None) | ||||
|         host_filter = choose_host_filter(filter_name) | ||||
|  | ||||
|         # TODO(sandy): We're only using InstanceType-based specs | ||||
|         # currently. Later we'll need to snoop for more detailed | ||||
|         # host filter requests. | ||||
|         instance_type = request_spec['instance_type'] | ||||
|         name, query = host_filter.instance_type_to_filter(instance_type) | ||||
|         return host_filter.filter_hosts(self.zone_manager, query) | ||||
|  | ||||
|     def weigh_hosts(self, num, request_spec, hosts): | ||||
|         """Derived classes must override this method and return | ||||
|         a lists of hosts in [{weight, hostname}] format. | ||||
|         """ | ||||
|         return [dict(weight=1, hostname=host) for host, caps in hosts] | ||||
|   | ||||
							
								
								
									
										156
									
								
								nova/scheduler/least_cost.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								nova/scheduler/least_cost.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| # Copyright (c) 2011 Openstack, LLC. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| #    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. | ||||
| """ | ||||
| Least Cost Scheduler is a mechanism for choosing which host machines to | ||||
| provision a set of resources to. The input of the least-cost-scheduler is a | ||||
| set of objective-functions, called the 'cost-functions', a weight for each | ||||
| cost-function, and a list of candidate hosts (gathered via FilterHosts). | ||||
|  | ||||
| The cost-function and weights are tabulated, and the host with the least cost | ||||
| is then selected for provisioning. | ||||
| """ | ||||
|  | ||||
| import collections | ||||
|  | ||||
| from nova import flags | ||||
| from nova import log as logging | ||||
| from nova.scheduler import zone_aware_scheduler | ||||
| from nova import utils | ||||
|  | ||||
| LOG = logging.getLogger('nova.scheduler.least_cost') | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
| flags.DEFINE_list('least_cost_scheduler_cost_functions', | ||||
|                   ['nova.scheduler.least_cost.noop_cost_fn'], | ||||
|                   'Which cost functions the LeastCostScheduler should use.') | ||||
|  | ||||
|  | ||||
| # TODO(sirp): Once we have enough of these rules, we can break them out into a | ||||
| # cost_functions.py file (perhaps in a least_cost_scheduler directory) | ||||
| flags.DEFINE_integer('noop_cost_fn_weight', 1, | ||||
|                      'How much weight to give the noop cost function') | ||||
|  | ||||
|  | ||||
| def noop_cost_fn(host): | ||||
|     """Return a pre-weight cost of 1 for each host""" | ||||
|     return 1 | ||||
|  | ||||
|  | ||||
| flags.DEFINE_integer('fill_first_cost_fn_weight', 1, | ||||
|                      'How much weight to give the fill-first cost function') | ||||
|  | ||||
|  | ||||
| def fill_first_cost_fn(host): | ||||
|     """Prefer hosts that have less ram available, filter_hosts will exclude | ||||
|     hosts that don't have enough ram""" | ||||
|     hostname, caps = host | ||||
|     free_mem = caps['compute']['host_memory_free'] | ||||
|     return free_mem | ||||
|  | ||||
|  | ||||
| class LeastCostScheduler(zone_aware_scheduler.ZoneAwareScheduler): | ||||
|     def get_cost_fns(self): | ||||
|         """Returns a list of tuples containing weights and cost functions to | ||||
|         use for weighing hosts | ||||
|         """ | ||||
|         cost_fns = [] | ||||
|         for cost_fn_str in FLAGS.least_cost_scheduler_cost_functions: | ||||
|  | ||||
|             try: | ||||
|                 # NOTE(sirp): import_class is somewhat misnamed since it can | ||||
|                 # any callable from a module | ||||
|                 cost_fn = utils.import_class(cost_fn_str) | ||||
|             except exception.ClassNotFound: | ||||
|                 raise exception.SchedulerCostFunctionNotFound( | ||||
|                     cost_fn_str=cost_fn_str) | ||||
|  | ||||
|             try: | ||||
|                 weight = getattr(FLAGS, "%s_weight" % cost_fn.__name__) | ||||
|             except AttributeError: | ||||
|                 raise exception.SchedulerWeightFlagNotFound( | ||||
|                     flag_name=flag_name) | ||||
|  | ||||
|             cost_fns.append((weight, cost_fn)) | ||||
|  | ||||
|         return cost_fns | ||||
|  | ||||
|     def weigh_hosts(self, num, request_spec, hosts): | ||||
|         """Returns a list of dictionaries of form: | ||||
|             [ {weight: weight, hostname: hostname} ]""" | ||||
|  | ||||
|         # FIXME(sirp): weigh_hosts should handle more than just instances | ||||
|         hostnames = [hostname for hostname, caps in hosts] | ||||
|  | ||||
|         cost_fns = self.get_cost_fns() | ||||
|         costs = weighted_sum(domain=hosts, weighted_fns=cost_fns) | ||||
|  | ||||
|         weighted = [] | ||||
|         weight_log = [] | ||||
|         for cost, hostname in zip(costs, hostnames): | ||||
|             weight_log.append("%s: %s" % (hostname, "%.2f" % cost)) | ||||
|             weight_dict = dict(weight=cost, hostname=hostname) | ||||
|             weighted.append(weight_dict) | ||||
|  | ||||
|         LOG.debug(_("Weighted Costs => %s") % weight_log) | ||||
|         return weighted | ||||
|  | ||||
|  | ||||
| def normalize_list(L): | ||||
|     """Normalize an array of numbers such that each element satisfies: | ||||
|         0 <= e <= 1""" | ||||
|     if not L: | ||||
|         return L | ||||
|     max_ = max(L) | ||||
|     if max_ > 0: | ||||
|         return [(float(e) / max_) for e in L] | ||||
|     return L | ||||
|  | ||||
|  | ||||
| def weighted_sum(domain, weighted_fns, normalize=True): | ||||
|     """Use the weighted-sum method to compute a score for an array of objects. | ||||
|     Normalize the results of the objective-functions so that the weights are | ||||
|     meaningful regardless of objective-function's range. | ||||
|  | ||||
|     domain - input to be scored | ||||
|     weighted_fns - list of weights and functions like: | ||||
|         [(weight, objective-functions)] | ||||
|  | ||||
|     Returns an unsorted of scores. To pair with hosts do: zip(scores, hosts) | ||||
|     """ | ||||
|     # Table of form: | ||||
|     #   { domain1: [score1, score2, ..., scoreM] | ||||
|     #     ... | ||||
|     #     domainN: [score1, score2, ..., scoreM] } | ||||
|     score_table = collections.defaultdict(list) | ||||
|     for weight, fn in weighted_fns: | ||||
|         scores = [fn(elem) for elem in domain] | ||||
|  | ||||
|         if normalize: | ||||
|             norm_scores = normalize_list(scores) | ||||
|         else: | ||||
|             norm_scores = scores | ||||
|  | ||||
|         for idx, score in enumerate(norm_scores): | ||||
|             weighted_score = score * weight | ||||
|             score_table[idx].append(weighted_score) | ||||
|  | ||||
|     # Sum rows in table to compute score for each element in domain | ||||
|     domain_scores = [] | ||||
|     for idx in sorted(score_table): | ||||
|         elem_score = sum(score_table[idx]) | ||||
|         elem = domain[idx] | ||||
|         domain_scores.append(elem_score) | ||||
|  | ||||
|     return domain_scores | ||||
| @@ -21,10 +21,9 @@ | ||||
| Simple Scheduler | ||||
| """ | ||||
|  | ||||
| import datetime | ||||
|  | ||||
| from nova import db | ||||
| from nova import flags | ||||
| from nova import utils | ||||
| from nova.scheduler import driver | ||||
| from nova.scheduler import chance | ||||
|  | ||||
| @@ -54,7 +53,7 @@ class SimpleScheduler(chance.ChanceScheduler): | ||||
|  | ||||
|             # TODO(vish): this probably belongs in the manager, if we | ||||
|             #             can generalize this somehow | ||||
|             now = datetime.datetime.utcnow() | ||||
|             now = utils.utcnow() | ||||
|             db.instance_update(context, instance_id, {'host': host, | ||||
|                                                       'scheduled_at': now}) | ||||
|             return host | ||||
| @@ -66,7 +65,7 @@ class SimpleScheduler(chance.ChanceScheduler): | ||||
|             if self.service_is_up(service): | ||||
|                 # NOTE(vish): this probably belongs in the manager, if we | ||||
|                 #             can generalize this somehow | ||||
|                 now = datetime.datetime.utcnow() | ||||
|                 now = utils.utcnow() | ||||
|                 db.instance_update(context, | ||||
|                                    instance_id, | ||||
|                                    {'host': service['host'], | ||||
| @@ -96,7 +95,7 @@ class SimpleScheduler(chance.ChanceScheduler): | ||||
|  | ||||
|             # TODO(vish): this probably belongs in the manager, if we | ||||
|             #             can generalize this somehow | ||||
|             now = datetime.datetime.utcnow() | ||||
|             now = utils.utcnow() | ||||
|             db.volume_update(context, volume_id, {'host': host, | ||||
|                                                   'scheduled_at': now}) | ||||
|             return host | ||||
| @@ -109,7 +108,7 @@ class SimpleScheduler(chance.ChanceScheduler): | ||||
|             if self.service_is_up(service): | ||||
|                 # NOTE(vish): this probably belongs in the manager, if we | ||||
|                 #             can generalize this somehow | ||||
|                 now = datetime.datetime.utcnow() | ||||
|                 now = utils.utcnow() | ||||
|                 db.volume_update(context, | ||||
|                                  volume_id, | ||||
|                                  {'host': service['host'], | ||||
|   | ||||
| @@ -21,14 +21,30 @@ across zones. There are two expansion points to this class for: | ||||
| """ | ||||
|  | ||||
| import operator | ||||
| import json | ||||
|  | ||||
| import M2Crypto | ||||
| import novaclient | ||||
|  | ||||
| from nova import crypto | ||||
| from nova import db | ||||
| from nova import exception | ||||
| from nova import flags | ||||
| from nova import log as logging | ||||
| from nova import rpc | ||||
|  | ||||
| from nova.scheduler import api | ||||
| from nova.scheduler import driver | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
| LOG = logging.getLogger('nova.scheduler.zone_aware_scheduler') | ||||
|  | ||||
|  | ||||
| class InvalidBlob(exception.NovaException): | ||||
|     message = _("Ill-formed or incorrectly routed 'blob' data sent " | ||||
|                 "to instance create request.") | ||||
|  | ||||
|  | ||||
| class ZoneAwareScheduler(driver.Scheduler): | ||||
|     """Base class for creating Zone Aware Schedulers.""" | ||||
|  | ||||
| @@ -36,84 +52,215 @@ class ZoneAwareScheduler(driver.Scheduler): | ||||
|         """Call novaclient zone method. Broken out for testing.""" | ||||
|         return api.call_zone_method(context, method, specs=specs) | ||||
|  | ||||
|     def schedule_run_instance(self, context, topic='compute', specs={}, | ||||
|                                         *args, **kwargs): | ||||
|     def _provision_resource_locally(self, context, item, instance_id, kwargs): | ||||
|         """Create the requested resource in this Zone.""" | ||||
|         host = item['hostname'] | ||||
|         kwargs['instance_id'] = instance_id | ||||
|         rpc.cast(context, | ||||
|                  db.queue_get_for(context, "compute", host), | ||||
|                  {"method": "run_instance", | ||||
|                   "args": kwargs}) | ||||
|         LOG.debug(_("Provisioning locally via compute node %(host)s") | ||||
|                             % locals()) | ||||
|  | ||||
|     def _decrypt_blob(self, blob): | ||||
|         """Returns the decrypted blob or None if invalid. Broken out | ||||
|         for testing.""" | ||||
|         decryptor = crypto.decryptor(FLAGS.build_plan_encryption_key) | ||||
|         try: | ||||
|             json_entry = decryptor(blob) | ||||
|             return json.dumps(entry) | ||||
|         except M2Crypto.EVP.EVPError: | ||||
|             pass | ||||
|         return None | ||||
|  | ||||
|     def _ask_child_zone_to_create_instance(self, context, zone_info, | ||||
|                                            request_spec, kwargs): | ||||
|         """Once we have determined that the request should go to one | ||||
|         of our children, we need to fabricate a new POST /servers/ | ||||
|         call with the same parameters that were passed into us. | ||||
|  | ||||
|         Note that we have to reverse engineer from our args to get back the | ||||
|         image, flavor, ipgroup, etc. since the original call could have | ||||
|         come in from EC2 (which doesn't use these things).""" | ||||
|  | ||||
|         instance_type = request_spec['instance_type'] | ||||
|         instance_properties = request_spec['instance_properties'] | ||||
|  | ||||
|         name = instance_properties['display_name'] | ||||
|         image_id = instance_properties['image_id'] | ||||
|         meta = instance_properties['metadata'] | ||||
|         flavor_id = instance_type['flavorid'] | ||||
|         reservation_id = instance_properties['reservation_id'] | ||||
|  | ||||
|         files = kwargs['injected_files'] | ||||
|         ipgroup = None  # Not supported in OS API ... yet | ||||
|  | ||||
|         child_zone = zone_info['child_zone'] | ||||
|         child_blob = zone_info['child_blob'] | ||||
|         zone = db.zone_get(context, child_zone) | ||||
|         url = zone.api_url | ||||
|         LOG.debug(_("Forwarding instance create call to child zone %(url)s" | ||||
|                     ". ReservationID=%(reservation_id)s") | ||||
|                     % locals()) | ||||
|         nova = None | ||||
|         try: | ||||
|             nova = novaclient.OpenStack(zone.username, zone.password, url) | ||||
|             nova.authenticate() | ||||
|         except novaclient.exceptions.BadRequest, e: | ||||
|             raise exception.NotAuthorized(_("Bad credentials attempting " | ||||
|                             "to talk to zone at %(url)s.") % locals()) | ||||
|  | ||||
|         nova.servers.create(name, image_id, flavor_id, ipgroup, meta, files, | ||||
|                             child_blob, reservation_id=reservation_id) | ||||
|  | ||||
|     def _provision_resource_from_blob(self, context, item, instance_id, | ||||
|                                           request_spec, kwargs): | ||||
|         """Create the requested resource locally or in a child zone | ||||
|            based on what is stored in the zone blob info. | ||||
|  | ||||
|            Attempt to decrypt the blob to see if this request is: | ||||
|            1. valid, and | ||||
|            2. intended for this zone or a child zone. | ||||
|  | ||||
|            Note: If we have "blob" that means the request was passed | ||||
|            into us from a parent zone. If we have "child_blob" that | ||||
|            means we gathered the info from one of our children. | ||||
|            It's possible that, when we decrypt the 'blob' field, it | ||||
|            contains "child_blob" data. In which case we forward the | ||||
|            request.""" | ||||
|  | ||||
|         host_info = None | ||||
|         if "blob" in item: | ||||
|             # Request was passed in from above. Is it for us? | ||||
|             host_info = self._decrypt_blob(item['blob']) | ||||
|         elif "child_blob" in item: | ||||
|             # Our immediate child zone provided this info ... | ||||
|             host_info = item | ||||
|  | ||||
|         if not host_info: | ||||
|             raise InvalidBlob() | ||||
|  | ||||
|         # Valid data ... is it for us? | ||||
|         if 'child_zone' in host_info and 'child_blob' in host_info: | ||||
|             self._ask_child_zone_to_create_instance(context, host_info, | ||||
|                                                     request_spec, kwargs) | ||||
|         else: | ||||
|             self._provision_resource_locally(context, host_info, | ||||
|                                              instance_id, kwargs) | ||||
|  | ||||
|     def _provision_resource(self, context, item, instance_id, request_spec, | ||||
|                            kwargs): | ||||
|         """Create the requested resource in this Zone or a child zone.""" | ||||
|         if "hostname" in item: | ||||
|             self._provision_resource_locally(context, item, instance_id, | ||||
|                             kwargs) | ||||
|             return | ||||
|  | ||||
|         self._provision_resource_from_blob(context, item, instance_id, | ||||
|                                                request_spec, kwargs) | ||||
|  | ||||
|     def schedule_run_instance(self, context, instance_id, request_spec, | ||||
|                               *args, **kwargs): | ||||
|         """This method is called from nova.compute.api to provision | ||||
|         an instance. However we need to look at the parameters being | ||||
|         passed in to see if this is a request to: | ||||
|         1. Create a Build Plan and then provision, or | ||||
|         2. Use the Build Plan information in the request parameters | ||||
|            to simply create the instance (either in this zone or | ||||
|            a child zone).""" | ||||
|            a child zone). | ||||
|         """ | ||||
|  | ||||
|         if 'blob' in specs: | ||||
|             return self.provision_instance(context, topic, specs) | ||||
|         # TODO(sandy): We'll have to look for richer specs at some point. | ||||
|  | ||||
|         blob = request_spec.get('blob') | ||||
|         if blob: | ||||
|             self._provision_resource(context, request_spec, instance_id, | ||||
|                                     request_spec, kwargs) | ||||
|             return None | ||||
|  | ||||
|         # Create build plan and provision ... | ||||
|         build_plan = self.select(context, specs) | ||||
|         build_plan = self.select(context, request_spec) | ||||
|         if not build_plan: | ||||
|             raise driver.NoValidHost(_('No hosts were available')) | ||||
|  | ||||
|         for item in build_plan: | ||||
|             self.provision_instance(context, topic, item) | ||||
|             self._provision_resource(context, item, instance_id, request_spec, | ||||
|                                     kwargs) | ||||
|  | ||||
|     def provision_instance(context, topic, item): | ||||
|         """Create the requested instance in this Zone or a child zone.""" | ||||
|         pass | ||||
|         # Returning None short-circuits the routing to Compute (since | ||||
|         # we've already done it here) | ||||
|         return None | ||||
|  | ||||
|     def select(self, context, *args, **kwargs): | ||||
|     def select(self, context, request_spec, *args, **kwargs): | ||||
|         """Select returns a list of weights and zone/host information | ||||
|         corresponding to the best hosts to service the request. Any | ||||
|         child zone information has been encrypted so as not to reveal | ||||
|         anything about the children.""" | ||||
|         return self._schedule(context, "compute", *args, **kwargs) | ||||
|         anything about the children. | ||||
|         """ | ||||
|         return self._schedule(context, "compute", request_spec, | ||||
|                               *args, **kwargs) | ||||
|  | ||||
|     def schedule(self, context, topic, *args, **kwargs): | ||||
|     # TODO(sandy): We're only focused on compute instances right now, | ||||
|     # so we don't implement the default "schedule()" method required | ||||
|     # of Schedulers. | ||||
|     def schedule(self, context, topic, request_spec, *args, **kwargs): | ||||
|         """The schedule() contract requires we return the one | ||||
|         best-suited host for this request. | ||||
|         """ | ||||
|         res = self._schedule(context, topic, *args, **kwargs) | ||||
|         # TODO(sirp): should this be a host object rather than a weight-dict? | ||||
|         if not res: | ||||
|             raise driver.NoValidHost(_('No hosts were available')) | ||||
|         return res[0] | ||||
|         raise driver.NoValidHost(_('No hosts were available')) | ||||
|  | ||||
|     def _schedule(self, context, topic, *args, **kwargs): | ||||
|     def _schedule(self, context, topic, request_spec, *args, **kwargs): | ||||
|         """Returns a list of hosts that meet the required specs, | ||||
|         ordered by their fitness. | ||||
|         """ | ||||
|  | ||||
|         #TODO(sandy): extract these from args. | ||||
|         if topic != "compute": | ||||
|             raise NotImplemented(_("Zone Aware Scheduler only understands " | ||||
|                                    "Compute nodes (for now)")) | ||||
|  | ||||
|         #TODO(sandy): how to infer this from OS API params? | ||||
|         num_instances = 1 | ||||
|         specs = {} | ||||
|  | ||||
|         # Filter local hosts based on requirements ... | ||||
|         host_list = self.filter_hosts(num_instances, specs) | ||||
|         host_list = self.filter_hosts(num_instances, request_spec) | ||||
|  | ||||
|         # TODO(sirp): weigh_hosts should also be a function of 'topic' or | ||||
|         # resources, so that we can apply different objective functions to it | ||||
|  | ||||
|         # then weigh the selected hosts. | ||||
|         # weighted = [{weight=weight, name=hostname}, ...] | ||||
|         weighted = self.weigh_hosts(num_instances, specs, host_list) | ||||
|         weighted = self.weigh_hosts(num_instances, request_spec, host_list) | ||||
|  | ||||
|         # Next, tack on the best weights from the child zones ... | ||||
|         json_spec = json.dumps(request_spec) | ||||
|         child_results = self._call_zone_method(context, "select", | ||||
|                 specs=specs) | ||||
|                 specs=json_spec) | ||||
|         for child_zone, result in child_results: | ||||
|             for weighting in result: | ||||
|                 # Remember the child_zone so we can get back to | ||||
|                 # it later if needed. This implicitly builds a zone | ||||
|                 # path structure. | ||||
|                 host_dict = { | ||||
|                         "weight": weighting["weight"], | ||||
|                         "child_zone": child_zone, | ||||
|                         "child_blob": weighting["blob"]} | ||||
|                 host_dict = {"weight": weighting["weight"], | ||||
|                              "child_zone": child_zone, | ||||
|                              "child_blob": weighting["blob"]} | ||||
|                 weighted.append(host_dict) | ||||
|  | ||||
|         weighted.sort(key=operator.itemgetter('weight')) | ||||
|         return weighted | ||||
|  | ||||
|     def filter_hosts(self, num, specs): | ||||
|     def filter_hosts(self, num, request_spec): | ||||
|         """Derived classes must override this method and return | ||||
|            a list of hosts in [(hostname, capability_dict)] format.""" | ||||
|         raise NotImplemented() | ||||
|            a list of hosts in [(hostname, capability_dict)] format. | ||||
|         """ | ||||
|         # NOTE(sirp): The default logic is the equivalent to AllHostsFilter | ||||
|         service_states = self.zone_manager.service_states | ||||
|         return [(host, services) | ||||
|                 for host, services in service_states.iteritems()] | ||||
|  | ||||
|     def weigh_hosts(self, num, specs, hosts): | ||||
|         """Derived classes must override this method and return | ||||
|            a lists of hosts in [{weight, hostname}] format.""" | ||||
|         raise NotImplemented() | ||||
|     def weigh_hosts(self, num, request_spec, hosts): | ||||
|         """Derived classes may override this to provide more sophisticated | ||||
|         scheduling objectives | ||||
|         """ | ||||
|         # NOTE(sirp): The default logic is the same as the NoopCostFunction | ||||
|         return [dict(weight=1, hostname=host) for host, caps in hosts] | ||||
|   | ||||
| @@ -17,16 +17,17 @@ | ||||
| ZoneManager oversees all communications with child Zones. | ||||
| """ | ||||
|  | ||||
| import datetime | ||||
| import novaclient | ||||
| import thread | ||||
| import traceback | ||||
|  | ||||
| from datetime import datetime | ||||
| from eventlet import greenpool | ||||
|  | ||||
| from nova import db | ||||
| from nova import flags | ||||
| from nova import log as logging | ||||
| from nova import utils | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
| flags.DEFINE_integer('zone_db_check_interval', 60, | ||||
| @@ -42,7 +43,7 @@ class ZoneState(object): | ||||
|         self.name = None | ||||
|         self.capabilities = None | ||||
|         self.attempt = 0 | ||||
|         self.last_seen = datetime.min | ||||
|         self.last_seen = datetime.datetime.min | ||||
|         self.last_exception = None | ||||
|         self.last_exception_time = None | ||||
|  | ||||
| @@ -56,7 +57,7 @@ class ZoneState(object): | ||||
|     def update_metadata(self, zone_metadata): | ||||
|         """Update zone metadata after successful communications with | ||||
|            child zone.""" | ||||
|         self.last_seen = datetime.now() | ||||
|         self.last_seen = utils.utcnow() | ||||
|         self.attempt = 0 | ||||
|         self.name = zone_metadata.get("name", "n/a") | ||||
|         self.capabilities = ", ".join(["%s=%s" % (k, v) | ||||
| @@ -72,7 +73,7 @@ class ZoneState(object): | ||||
|         """Something went wrong. Check to see if zone should be | ||||
|            marked as offline.""" | ||||
|         self.last_exception = exception | ||||
|         self.last_exception_time = datetime.now() | ||||
|         self.last_exception_time = utils.utcnow() | ||||
|         api_url = self.api_url | ||||
|         logging.warning(_("'%(exception)s' error talking to " | ||||
|                           "zone %(api_url)s") % locals()) | ||||
| @@ -104,7 +105,7 @@ def _poll_zone(zone): | ||||
| class ZoneManager(object): | ||||
|     """Keeps the zone states updated.""" | ||||
|     def __init__(self): | ||||
|         self.last_zone_db_check = datetime.min | ||||
|         self.last_zone_db_check = datetime.datetime.min | ||||
|         self.zone_states = {}  # { <zone_id> : ZoneState } | ||||
|         self.service_states = {}  # { <host> : { <service> : { cap k : v }}} | ||||
|         self.green_pool = greenpool.GreenPool() | ||||
| @@ -158,10 +159,10 @@ class ZoneManager(object): | ||||
|  | ||||
|     def ping(self, context=None): | ||||
|         """Ping should be called periodically to update zone status.""" | ||||
|         diff = datetime.now() - self.last_zone_db_check | ||||
|         diff = utils.utcnow() - self.last_zone_db_check | ||||
|         if diff.seconds >= FLAGS.zone_db_check_interval: | ||||
|             logging.debug(_("Updating zone cache from db.")) | ||||
|             self.last_zone_db_check = datetime.now() | ||||
|             self.last_zone_db_check = utils.utcnow() | ||||
|             self._refresh_from_db(context) | ||||
|         self._poll_zones(context) | ||||
|  | ||||
|   | ||||
							
								
								
									
										0
									
								
								nova/tests/scheduler/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								nova/tests/scheduler/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										206
									
								
								nova/tests/scheduler/test_host_filter.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										206
									
								
								nova/tests/scheduler/test_host_filter.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,206 @@ | ||||
| # Copyright 2011 OpenStack LLC. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| #    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. | ||||
| """ | ||||
| Tests For Scheduler Host Filters. | ||||
| """ | ||||
|  | ||||
| import json | ||||
|  | ||||
| from nova import exception | ||||
| from nova import flags | ||||
| from nova import test | ||||
| from nova.scheduler import host_filter | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
|  | ||||
|  | ||||
| class FakeZoneManager: | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class HostFilterTestCase(test.TestCase): | ||||
|     """Test case for host filters.""" | ||||
|  | ||||
|     def _host_caps(self, multiplier): | ||||
|         # Returns host capabilities in the following way: | ||||
|         # host1 = memory:free 10 (100max) | ||||
|         #         disk:available 100 (1000max) | ||||
|         # hostN = memory:free 10 + 10N | ||||
|         #         disk:available 100 + 100N | ||||
|         # in other words: hostN has more resources than host0 | ||||
|         # which means ... don't go above 10 hosts. | ||||
|         return {'host_name-description': 'XenServer %s' % multiplier, | ||||
|                 'host_hostname': 'xs-%s' % multiplier, | ||||
|                 'host_memory_total': 100, | ||||
|                 'host_memory_overhead': 10, | ||||
|                 'host_memory_free': 10 + multiplier * 10, | ||||
|                 'host_memory_free-computed': 10 + multiplier * 10, | ||||
|                 'host_other-config': {}, | ||||
|                 'host_ip_address': '192.168.1.%d' % (100 + multiplier), | ||||
|                 'host_cpu_info': {}, | ||||
|                 'disk_available': 100 + multiplier * 100, | ||||
|                 'disk_total': 1000, | ||||
|                 'disk_used': 0, | ||||
|                 'host_uuid': 'xxx-%d' % multiplier, | ||||
|                 'host_name-label': 'xs-%s' % multiplier} | ||||
|  | ||||
|     def setUp(self): | ||||
|         self.old_flag = FLAGS.default_host_filter | ||||
|         FLAGS.default_host_filter = \ | ||||
|                             'nova.scheduler.host_filter.AllHostsFilter' | ||||
|         self.instance_type = dict(name='tiny', | ||||
|                 memory_mb=50, | ||||
|                 vcpus=10, | ||||
|                 local_gb=500, | ||||
|                 flavorid=1, | ||||
|                 swap=500, | ||||
|                 rxtx_quota=30000, | ||||
|                 rxtx_cap=200) | ||||
|  | ||||
|         self.zone_manager = FakeZoneManager() | ||||
|         states = {} | ||||
|         for x in xrange(10): | ||||
|             states['host%02d' % (x + 1)] = {'compute': self._host_caps(x)} | ||||
|         self.zone_manager.service_states = states | ||||
|  | ||||
|     def tearDown(self): | ||||
|         FLAGS.default_host_filter = self.old_flag | ||||
|  | ||||
|     def test_choose_filter(self): | ||||
|         # Test default filter ... | ||||
|         hf = host_filter.choose_host_filter() | ||||
|         self.assertEquals(hf._full_name(), | ||||
|                         'nova.scheduler.host_filter.AllHostsFilter') | ||||
|         # Test valid filter ... | ||||
|         hf = host_filter.choose_host_filter( | ||||
|                         'nova.scheduler.host_filter.InstanceTypeFilter') | ||||
|         self.assertEquals(hf._full_name(), | ||||
|                         'nova.scheduler.host_filter.InstanceTypeFilter') | ||||
|         # Test invalid filter ... | ||||
|         try: | ||||
|             host_filter.choose_host_filter('does not exist') | ||||
|             self.fail("Should not find host filter.") | ||||
|         except exception.SchedulerHostFilterNotFound: | ||||
|             pass | ||||
|  | ||||
|     def test_all_host_filter(self): | ||||
|         hf = host_filter.AllHostsFilter() | ||||
|         cooked = hf.instance_type_to_filter(self.instance_type) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|         self.assertEquals(10, len(hosts)) | ||||
|         for host, capabilities in hosts: | ||||
|             self.assertTrue(host.startswith('host')) | ||||
|  | ||||
|     def test_instance_type_filter(self): | ||||
|         hf = host_filter.InstanceTypeFilter() | ||||
|         # filter all hosts that can support 50 ram and 500 disk | ||||
|         name, cooked = hf.instance_type_to_filter(self.instance_type) | ||||
|         self.assertEquals('nova.scheduler.host_filter.InstanceTypeFilter', | ||||
|                           name) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|         self.assertEquals(6, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
|         self.assertEquals('host05', just_hosts[0]) | ||||
|         self.assertEquals('host10', just_hosts[5]) | ||||
|  | ||||
|     def test_json_filter(self): | ||||
|         hf = host_filter.JsonFilter() | ||||
|         # filter all hosts that can support 50 ram and 500 disk | ||||
|         name, cooked = hf.instance_type_to_filter(self.instance_type) | ||||
|         self.assertEquals('nova.scheduler.host_filter.JsonFilter', name) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|         self.assertEquals(6, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
|         self.assertEquals('host05', just_hosts[0]) | ||||
|         self.assertEquals('host10', just_hosts[5]) | ||||
|  | ||||
|         # Try some custom queries | ||||
|  | ||||
|         raw = ['or', | ||||
|                    ['and', | ||||
|                        ['<', '$compute.host_memory_free', 30], | ||||
|                        ['<', '$compute.disk_available', 300], | ||||
|                    ], | ||||
|                    ['and', | ||||
|                        ['>', '$compute.host_memory_free', 70], | ||||
|                        ['>', '$compute.disk_available', 700], | ||||
|                    ] | ||||
|               ] | ||||
|         cooked = json.dumps(raw) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|  | ||||
|         self.assertEquals(5, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
|         for index, host in zip([1, 2, 8, 9, 10], just_hosts): | ||||
|             self.assertEquals('host%02d' % index, host) | ||||
|  | ||||
|         raw = ['not', | ||||
|                   ['=', '$compute.host_memory_free', 30], | ||||
|               ] | ||||
|         cooked = json.dumps(raw) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|  | ||||
|         self.assertEquals(9, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
|         for index, host in zip([1, 2, 4, 5, 6, 7, 8, 9, 10], just_hosts): | ||||
|             self.assertEquals('host%02d' % index, host) | ||||
|  | ||||
|         raw = ['in', '$compute.host_memory_free', 20, 40, 60, 80, 100] | ||||
|         cooked = json.dumps(raw) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|  | ||||
|         self.assertEquals(5, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
|         for index, host in zip([2, 4, 6, 8, 10], just_hosts): | ||||
|             self.assertEquals('host%02d' % index, host) | ||||
|  | ||||
|         # Try some bogus input ... | ||||
|         raw = ['unknown command', ] | ||||
|         cooked = json.dumps(raw) | ||||
|         try: | ||||
|             hf.filter_hosts(self.zone_manager, cooked) | ||||
|             self.fail("Should give KeyError") | ||||
|         except KeyError, e: | ||||
|             pass | ||||
|  | ||||
|         self.assertTrue(hf.filter_hosts(self.zone_manager, json.dumps([]))) | ||||
|         self.assertTrue(hf.filter_hosts(self.zone_manager, json.dumps({}))) | ||||
|         self.assertTrue(hf.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 ['not', True, False, True, False], | ||||
|             ))) | ||||
|  | ||||
|         try: | ||||
|             hf.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 'not', True, False, True, False, | ||||
|             )) | ||||
|             self.fail("Should give KeyError") | ||||
|         except KeyError, e: | ||||
|             pass | ||||
|  | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps(['=', '$foo', 100]))) | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps(['=', '$.....', 100]))) | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps( | ||||
|             ['>', ['and', ['or', ['not', ['<', ['>=', ['<=', ['in', ]]]]]]]]))) | ||||
|  | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps(['=', {}, ['>', '$missing....foo']]))) | ||||
							
								
								
									
										144
									
								
								nova/tests/scheduler/test_least_cost_scheduler.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								nova/tests/scheduler/test_least_cost_scheduler.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,144 @@ | ||||
| # Copyright 2011 OpenStack LLC. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| #    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. | ||||
| """ | ||||
| Tests For Least Cost Scheduler | ||||
| """ | ||||
|  | ||||
| from nova import flags | ||||
| from nova import test | ||||
| from nova.scheduler import least_cost | ||||
| from nova.tests.scheduler import test_zone_aware_scheduler | ||||
|  | ||||
| MB = 1024 * 1024 | ||||
| FLAGS = flags.FLAGS | ||||
|  | ||||
|  | ||||
| class FakeHost(object): | ||||
|     def __init__(self, host_id, free_ram, io): | ||||
|         self.id = host_id | ||||
|         self.free_ram = free_ram | ||||
|         self.io = io | ||||
|  | ||||
|  | ||||
| class WeightedSumTestCase(test.TestCase): | ||||
|     def test_empty_domain(self): | ||||
|         domain = [] | ||||
|         weighted_fns = [] | ||||
|         result = least_cost.weighted_sum(domain, weighted_fns) | ||||
|         expected = [] | ||||
|         self.assertEqual(expected, result) | ||||
|  | ||||
|     def test_basic_costing(self): | ||||
|         hosts = [ | ||||
|             FakeHost(1, 512 * MB, 100), | ||||
|             FakeHost(2, 256 * MB, 400), | ||||
|             FakeHost(3, 512 * MB, 100), | ||||
|         ] | ||||
|  | ||||
|         weighted_fns = [ | ||||
|             (1, lambda h: h.free_ram),  # Fill-first, free_ram is a *cost* | ||||
|             (2, lambda h: h.io),  # Avoid high I/O | ||||
|         ] | ||||
|  | ||||
|         costs = least_cost.weighted_sum( | ||||
|             domain=hosts, weighted_fns=weighted_fns) | ||||
|  | ||||
|         # Each 256 MB unit of free-ram contributes 0.5 points by way of: | ||||
|         #   cost = weight * (score/max_score) = 1 * (256/512) = 0.5 | ||||
|         # Each 100 iops of IO adds 0.5 points by way of: | ||||
|         #   cost = 2 * (100/400) = 2 * 0.25 = 0.5 | ||||
|         expected = [1.5, 2.5, 1.5] | ||||
|         self.assertEqual(expected, costs) | ||||
|  | ||||
|  | ||||
| class LeastCostSchedulerTestCase(test.TestCase): | ||||
|     def setUp(self): | ||||
|         super(LeastCostSchedulerTestCase, self).setUp() | ||||
|  | ||||
|         class FakeZoneManager: | ||||
|             pass | ||||
|  | ||||
|         zone_manager = FakeZoneManager() | ||||
|  | ||||
|         states = test_zone_aware_scheduler.fake_zone_manager_service_states( | ||||
|             num_hosts=10) | ||||
|         zone_manager.service_states = states | ||||
|  | ||||
|         self.sched = least_cost.LeastCostScheduler() | ||||
|         self.sched.zone_manager = zone_manager | ||||
|  | ||||
|     def tearDown(self): | ||||
|         super(LeastCostSchedulerTestCase, self).tearDown() | ||||
|  | ||||
|     def assertWeights(self, expected, num, request_spec, hosts): | ||||
|         weighted = self.sched.weigh_hosts(num, request_spec, hosts) | ||||
|         self.assertDictListMatch(weighted, expected, approx_equal=True) | ||||
|  | ||||
|     def test_no_hosts(self): | ||||
|         num = 1 | ||||
|         request_spec = {} | ||||
|         hosts = [] | ||||
|  | ||||
|         expected = [] | ||||
|         self.assertWeights(expected, num, request_spec, hosts) | ||||
|  | ||||
|     def test_noop_cost_fn(self): | ||||
|         FLAGS.least_cost_scheduler_cost_functions = [ | ||||
|             'nova.scheduler.least_cost.noop_cost_fn', | ||||
|         ] | ||||
|         FLAGS.noop_cost_fn_weight = 1 | ||||
|  | ||||
|         num = 1 | ||||
|         request_spec = {} | ||||
|         hosts = self.sched.filter_hosts(num, request_spec) | ||||
|  | ||||
|         expected = [dict(weight=1, hostname=hostname) | ||||
|                     for hostname, caps in hosts] | ||||
|         self.assertWeights(expected, num, request_spec, hosts) | ||||
|  | ||||
|     def test_cost_fn_weights(self): | ||||
|         FLAGS.least_cost_scheduler_cost_functions = [ | ||||
|             'nova.scheduler.least_cost.noop_cost_fn', | ||||
|         ] | ||||
|         FLAGS.noop_cost_fn_weight = 2 | ||||
|  | ||||
|         num = 1 | ||||
|         request_spec = {} | ||||
|         hosts = self.sched.filter_hosts(num, request_spec) | ||||
|  | ||||
|         expected = [dict(weight=2, hostname=hostname) | ||||
|                     for hostname, caps in hosts] | ||||
|         self.assertWeights(expected, num, request_spec, hosts) | ||||
|  | ||||
|     def test_fill_first_cost_fn(self): | ||||
|         FLAGS.least_cost_scheduler_cost_functions = [ | ||||
|             'nova.scheduler.least_cost.fill_first_cost_fn', | ||||
|         ] | ||||
|         FLAGS.fill_first_cost_fn_weight = 1 | ||||
|  | ||||
|         num = 1 | ||||
|         request_spec = {} | ||||
|         hosts = self.sched.filter_hosts(num, request_spec) | ||||
|  | ||||
|         expected = [] | ||||
|         for idx, (hostname, caps) in enumerate(hosts): | ||||
|             # Costs are normalized so over 10 hosts, each host with increasing | ||||
|             # free ram will cost 1/N more. Since the lowest cost host has some | ||||
|             # free ram, we add in the 1/N for the base_cost | ||||
|             weight = 0.1 + (0.1 * idx) | ||||
|             weight_dict = dict(weight=weight, hostname=hostname) | ||||
|             expected.append(weight_dict) | ||||
|  | ||||
|         self.assertWeights(expected, num, request_spec, hosts) | ||||
							
								
								
									
										296
									
								
								nova/tests/scheduler/test_zone_aware_scheduler.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										296
									
								
								nova/tests/scheduler/test_zone_aware_scheduler.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,296 @@ | ||||
| # Copyright 2011 OpenStack LLC. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| #    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. | ||||
| """ | ||||
| Tests For Zone Aware Scheduler. | ||||
| """ | ||||
|  | ||||
| from nova import exception | ||||
| from nova import test | ||||
| from nova.scheduler import driver | ||||
| from nova.scheduler import zone_aware_scheduler | ||||
| from nova.scheduler import zone_manager | ||||
|  | ||||
|  | ||||
| def _host_caps(multiplier): | ||||
|     # Returns host capabilities in the following way: | ||||
|     # host1 = memory:free 10 (100max) | ||||
|     #         disk:available 100 (1000max) | ||||
|     # hostN = memory:free 10 + 10N | ||||
|     #         disk:available 100 + 100N | ||||
|     # in other words: hostN has more resources than host0 | ||||
|     # which means ... don't go above 10 hosts. | ||||
|     return {'host_name-description': 'XenServer %s' % multiplier, | ||||
|             'host_hostname': 'xs-%s' % multiplier, | ||||
|             'host_memory_total': 100, | ||||
|             'host_memory_overhead': 10, | ||||
|             'host_memory_free': 10 + multiplier * 10, | ||||
|             'host_memory_free-computed': 10 + multiplier * 10, | ||||
|             'host_other-config': {}, | ||||
|             'host_ip_address': '192.168.1.%d' % (100 + multiplier), | ||||
|             'host_cpu_info': {}, | ||||
|             'disk_available': 100 + multiplier * 100, | ||||
|             'disk_total': 1000, | ||||
|             'disk_used': 0, | ||||
|             'host_uuid': 'xxx-%d' % multiplier, | ||||
|             'host_name-label': 'xs-%s' % multiplier} | ||||
|  | ||||
|  | ||||
| def fake_zone_manager_service_states(num_hosts): | ||||
|     states = {} | ||||
|     for x in xrange(num_hosts): | ||||
|         states['host%02d' % (x + 1)] = {'compute': _host_caps(x)} | ||||
|     return states | ||||
|  | ||||
|  | ||||
| class FakeZoneAwareScheduler(zone_aware_scheduler.ZoneAwareScheduler): | ||||
|     def filter_hosts(self, num, specs): | ||||
|         # NOTE(sirp): this is returning [(hostname, services)] | ||||
|         return self.zone_manager.service_states.items() | ||||
|  | ||||
|     def weigh_hosts(self, num, specs, hosts): | ||||
|         fake_weight = 99 | ||||
|         weighted = [] | ||||
|         for hostname, caps in hosts: | ||||
|             weighted.append(dict(weight=fake_weight, name=hostname)) | ||||
|         return weighted | ||||
|  | ||||
|  | ||||
| class FakeZoneManager(zone_manager.ZoneManager): | ||||
|     def __init__(self): | ||||
|         self.service_states = { | ||||
|             'host1': { | ||||
|                 'compute': {'ram': 1000}, | ||||
|             }, | ||||
|             'host2': { | ||||
|                 'compute': {'ram': 2000}, | ||||
|             }, | ||||
|             'host3': { | ||||
|                 'compute': {'ram': 3000}, | ||||
|             }, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class FakeEmptyZoneManager(zone_manager.ZoneManager): | ||||
|     def __init__(self): | ||||
|         self.service_states = {} | ||||
|  | ||||
|  | ||||
| def fake_empty_call_zone_method(context, method, specs): | ||||
|     return [] | ||||
|  | ||||
|  | ||||
| # Hmm, I should probably be using mox for this. | ||||
| was_called = False | ||||
|  | ||||
|  | ||||
| def fake_provision_resource(context, item, instance_id, request_spec, kwargs): | ||||
|     global was_called | ||||
|     was_called = True | ||||
|  | ||||
|  | ||||
| def fake_ask_child_zone_to_create_instance(context, zone_info, | ||||
|                                            request_spec, kwargs): | ||||
|     global was_called | ||||
|     was_called = True | ||||
|  | ||||
|  | ||||
| def fake_provision_resource_locally(context, item, instance_id, kwargs): | ||||
|     global was_called | ||||
|     was_called = True | ||||
|  | ||||
|  | ||||
| def fake_provision_resource_from_blob(context, item, instance_id, | ||||
|                                       request_spec, kwargs): | ||||
|     global was_called | ||||
|     was_called = True | ||||
|  | ||||
|  | ||||
| def fake_decrypt_blob_returns_local_info(blob): | ||||
|     return {'foo': True}  # values aren't important. | ||||
|  | ||||
|  | ||||
| def fake_decrypt_blob_returns_child_info(blob): | ||||
|     return {'child_zone': True, | ||||
|             'child_blob': True}  # values aren't important. Keys are. | ||||
|  | ||||
|  | ||||
| def fake_call_zone_method(context, method, specs): | ||||
|     return [ | ||||
|         ('zone1', [ | ||||
|             dict(weight=1, blob='AAAAAAA'), | ||||
|             dict(weight=111, blob='BBBBBBB'), | ||||
|             dict(weight=112, blob='CCCCCCC'), | ||||
|             dict(weight=113, blob='DDDDDDD'), | ||||
|         ]), | ||||
|         ('zone2', [ | ||||
|             dict(weight=120, blob='EEEEEEE'), | ||||
|             dict(weight=2, blob='FFFFFFF'), | ||||
|             dict(weight=122, blob='GGGGGGG'), | ||||
|             dict(weight=123, blob='HHHHHHH'), | ||||
|         ]), | ||||
|         ('zone3', [ | ||||
|             dict(weight=130, blob='IIIIIII'), | ||||
|             dict(weight=131, blob='JJJJJJJ'), | ||||
|             dict(weight=132, blob='KKKKKKK'), | ||||
|             dict(weight=3, blob='LLLLLLL'), | ||||
|         ]), | ||||
|     ] | ||||
|  | ||||
|  | ||||
| class ZoneAwareSchedulerTestCase(test.TestCase): | ||||
|     """Test case for Zone Aware Scheduler.""" | ||||
|  | ||||
|     def test_zone_aware_scheduler(self): | ||||
|         """ | ||||
|         Create a nested set of FakeZones, ensure that a select call returns the | ||||
|         appropriate build plan. | ||||
|         """ | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         self.stubs.Set(sched, '_call_zone_method', fake_call_zone_method) | ||||
|  | ||||
|         zm = FakeZoneManager() | ||||
|         sched.set_zone_manager(zm) | ||||
|  | ||||
|         fake_context = {} | ||||
|         build_plan = sched.select(fake_context, {}) | ||||
|  | ||||
|         self.assertEqual(15, len(build_plan)) | ||||
|  | ||||
|         hostnames = [plan_item['name'] | ||||
|                      for plan_item in build_plan if 'name' in plan_item] | ||||
|         self.assertEqual(3, len(hostnames)) | ||||
|  | ||||
|     def test_empty_zone_aware_scheduler(self): | ||||
|         """ | ||||
|         Ensure empty hosts & child_zones result in NoValidHosts exception. | ||||
|         """ | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         self.stubs.Set(sched, '_call_zone_method', fake_empty_call_zone_method) | ||||
|  | ||||
|         zm = FakeEmptyZoneManager() | ||||
|         sched.set_zone_manager(zm) | ||||
|  | ||||
|         fake_context = {} | ||||
|         self.assertRaises(driver.NoValidHost, sched.schedule_run_instance, | ||||
|                           fake_context, 1, | ||||
|                           dict(host_filter=None, | ||||
|                                request_spec={'instance_type': {}})) | ||||
|  | ||||
|     def test_schedule_do_not_schedule_with_hint(self): | ||||
|         """ | ||||
|         Check the local/child zone routing in the run_instance() call. | ||||
|         If the zone_blob hint was passed in, don't re-schedule. | ||||
|         """ | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         was_called = False | ||||
|         self.stubs.Set(sched, '_provision_resource', fake_provision_resource) | ||||
|         request_spec = { | ||||
|                 'instance_properties': {}, | ||||
|                 'instance_type': {}, | ||||
|                 'filter_driver': 'nova.scheduler.host_filter.AllHostsFilter', | ||||
|                 'blob': "Non-None blob data", | ||||
|             } | ||||
|  | ||||
|         result = sched.schedule_run_instance(None, 1, request_spec) | ||||
|         self.assertEquals(None, result) | ||||
|         self.assertTrue(was_called) | ||||
|  | ||||
|     def test_provision_resource_local(self): | ||||
|         """Provision a resource locally or remotely.""" | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         was_called = False | ||||
|         self.stubs.Set(sched, '_provision_resource_locally', | ||||
|                        fake_provision_resource_locally) | ||||
|  | ||||
|         request_spec = {'hostname': "foo"} | ||||
|         sched._provision_resource(None, request_spec, 1, request_spec, {}) | ||||
|         self.assertTrue(was_called) | ||||
|  | ||||
|     def test_provision_resource_remote(self): | ||||
|         """Provision a resource locally or remotely.""" | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         was_called = False | ||||
|         self.stubs.Set(sched, '_provision_resource_from_blob', | ||||
|                        fake_provision_resource_from_blob) | ||||
|  | ||||
|         request_spec = {} | ||||
|         sched._provision_resource(None, request_spec, 1, request_spec, {}) | ||||
|         self.assertTrue(was_called) | ||||
|  | ||||
|     def test_provision_resource_from_blob_empty(self): | ||||
|         """Provision a resource locally or remotely given no hints.""" | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         request_spec = {} | ||||
|         self.assertRaises(zone_aware_scheduler.InvalidBlob, | ||||
|                           sched._provision_resource_from_blob, | ||||
|                           None, {}, 1, {}, {}) | ||||
|  | ||||
|     def test_provision_resource_from_blob_with_local_blob(self): | ||||
|         """ | ||||
|         Provision a resource locally or remotely when blob hint passed in. | ||||
|         """ | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         was_called = False | ||||
|         self.stubs.Set(sched, '_decrypt_blob', | ||||
|                        fake_decrypt_blob_returns_local_info) | ||||
|         self.stubs.Set(sched, '_provision_resource_locally', | ||||
|                        fake_provision_resource_locally) | ||||
|  | ||||
|         request_spec = {'blob': "Non-None blob data"} | ||||
|  | ||||
|         sched._provision_resource_from_blob(None, request_spec, 1, | ||||
|                                             request_spec, {}) | ||||
|         self.assertTrue(was_called) | ||||
|  | ||||
|     def test_provision_resource_from_blob_with_child_blob(self): | ||||
|         """ | ||||
|         Provision a resource locally or remotely when child blob hint | ||||
|         passed in. | ||||
|         """ | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         self.stubs.Set(sched, '_decrypt_blob', | ||||
|                        fake_decrypt_blob_returns_child_info) | ||||
|         was_called = False | ||||
|         self.stubs.Set(sched, '_ask_child_zone_to_create_instance', | ||||
|                        fake_ask_child_zone_to_create_instance) | ||||
|  | ||||
|         request_spec = {'blob': "Non-None blob data"} | ||||
|  | ||||
|         sched._provision_resource_from_blob(None, request_spec, 1, | ||||
|                                             request_spec, {}) | ||||
|         self.assertTrue(was_called) | ||||
|  | ||||
|     def test_provision_resource_from_blob_with_immediate_child_blob(self): | ||||
|         """ | ||||
|         Provision a resource locally or remotely when blob hint passed in | ||||
|         from an immediate child. | ||||
|         """ | ||||
|         global was_called | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         was_called = False | ||||
|         self.stubs.Set(sched, '_ask_child_zone_to_create_instance', | ||||
|                        fake_ask_child_zone_to_create_instance) | ||||
|  | ||||
|         request_spec = {'child_blob': True, 'child_zone': True} | ||||
|  | ||||
|         sched._provision_resource_from_blob(None, request_spec, 1, | ||||
|                                             request_spec, {}) | ||||
|         self.assertTrue(was_called) | ||||
| @@ -86,6 +86,7 @@ class _AuthManagerBaseTestCase(test.TestCase): | ||||
|         super(_AuthManagerBaseTestCase, self).setUp() | ||||
|         self.flags(connection_type='fake') | ||||
|         self.manager = manager.AuthManager(new=True) | ||||
|         self.manager.mc.cache = {} | ||||
|  | ||||
|     def test_create_and_find_user(self): | ||||
|         with user_generator(self.manager): | ||||
|   | ||||
| @@ -26,17 +26,16 @@ from eventlet import greenthread | ||||
| from nova import context | ||||
| from nova import crypto | ||||
| from nova import db | ||||
| from nova import exception | ||||
| from nova import flags | ||||
| from nova import log as logging | ||||
| from nova import rpc | ||||
| from nova import test | ||||
| from nova import utils | ||||
| from nova import exception | ||||
| from nova.auth import manager | ||||
| from nova.api.ec2 import cloud | ||||
| from nova.api.ec2 import ec2utils | ||||
| from nova.image import local | ||||
| from nova.exception import NotFound | ||||
|  | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
| @@ -69,7 +68,7 @@ class CloudTestCase(test.TestCase): | ||||
|  | ||||
|         def fake_show(meh, context, id): | ||||
|             return {'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1, | ||||
|                     'type': 'machine'}} | ||||
|                     'type': 'machine', 'image_state': 'available'}} | ||||
|  | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show) | ||||
|         self.stubs.Set(local.LocalImageService, 'show_by_name', fake_show) | ||||
| @@ -117,6 +116,18 @@ class CloudTestCase(test.TestCase): | ||||
|                                   public_ip=address) | ||||
|         db.floating_ip_destroy(self.context, address) | ||||
|  | ||||
|     def test_allocate_address(self): | ||||
|         address = "10.10.10.10" | ||||
|         allocate = self.cloud.allocate_address | ||||
|         db.floating_ip_create(self.context, | ||||
|                               {'address': address, | ||||
|                                'host': self.network.host}) | ||||
|         self.assertEqual(allocate(self.context)['publicIp'], address) | ||||
|         db.floating_ip_destroy(self.context, address) | ||||
|         self.assertRaises(exception.NoMoreFloatingIps, | ||||
|                           allocate, | ||||
|                           self.context) | ||||
|  | ||||
|     def test_associate_disassociate_address(self): | ||||
|         """Verifies associate runs cleanly without raising an exception""" | ||||
|         address = "10.10.10.10" | ||||
| @@ -255,10 +266,10 @@ class CloudTestCase(test.TestCase): | ||||
|     def test_describe_instances(self): | ||||
|         """Makes sure describe_instances works and filters results.""" | ||||
|         inst1 = db.instance_create(self.context, {'reservation_id': 'a', | ||||
|                                                   'image_id': 1, | ||||
|                                                   'image_ref': 1, | ||||
|                                                   'host': 'host1'}) | ||||
|         inst2 = db.instance_create(self.context, {'reservation_id': 'a', | ||||
|                                                   'image_id': 1, | ||||
|                                                   'image_ref': 1, | ||||
|                                                   'host': 'host2'}) | ||||
|         comp1 = db.service_create(self.context, {'host': 'host1', | ||||
|                                                  'availability_zone': 'zone1', | ||||
| @@ -291,7 +302,7 @@ class CloudTestCase(test.TestCase): | ||||
|                     'type': 'machine'}}] | ||||
|  | ||||
|         def fake_show_none(meh, context, id): | ||||
|             raise NotFound | ||||
|             raise exception.ImageNotFound(image_id='bad_image_id') | ||||
|  | ||||
|         self.stubs.Set(local.LocalImageService, 'detail', fake_detail) | ||||
|         # list all | ||||
| @@ -309,7 +320,7 @@ class CloudTestCase(test.TestCase): | ||||
|         self.stubs.UnsetAll() | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show_none) | ||||
|         self.stubs.Set(local.LocalImageService, 'show_by_name', fake_show_none) | ||||
|         self.assertRaises(NotFound, describe_images, | ||||
|         self.assertRaises(exception.ImageNotFound, describe_images, | ||||
|                           self.context, ['ami-fake']) | ||||
|  | ||||
|     def test_describe_image_attribute(self): | ||||
| @@ -451,9 +462,67 @@ class CloudTestCase(test.TestCase): | ||||
|         self._create_key('test') | ||||
|         self.cloud.delete_key_pair(self.context, 'test') | ||||
|  | ||||
|     def test_run_instances(self): | ||||
|         kwargs = {'image_id': FLAGS.default_image, | ||||
|                   'instance_type': FLAGS.default_instance_type, | ||||
|                   'max_count': 1} | ||||
|         run_instances = self.cloud.run_instances | ||||
|         result = run_instances(self.context, **kwargs) | ||||
|         instance = result['instancesSet'][0] | ||||
|         self.assertEqual(instance['imageId'], 'ami-00000001') | ||||
|         self.assertEqual(instance['displayName'], 'Server 1') | ||||
|         self.assertEqual(instance['instanceId'], 'i-00000001') | ||||
|         self.assertEqual(instance['instanceState']['name'], 'networking') | ||||
|         self.assertEqual(instance['instanceType'], 'm1.small') | ||||
|  | ||||
|     def test_run_instances_image_state_none(self): | ||||
|         kwargs = {'image_id': FLAGS.default_image, | ||||
|                   'instance_type': FLAGS.default_instance_type, | ||||
|                   'max_count': 1} | ||||
|         run_instances = self.cloud.run_instances | ||||
|  | ||||
|         def fake_show_no_state(self, context, id): | ||||
|             return {'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1, | ||||
|                     'type': 'machine'}} | ||||
|  | ||||
|         self.stubs.UnsetAll() | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show_no_state) | ||||
|         self.assertRaises(exception.ApiError, run_instances, | ||||
|                           self.context, **kwargs) | ||||
|  | ||||
|     def test_run_instances_image_state_invalid(self): | ||||
|         kwargs = {'image_id': FLAGS.default_image, | ||||
|                   'instance_type': FLAGS.default_instance_type, | ||||
|                   'max_count': 1} | ||||
|         run_instances = self.cloud.run_instances | ||||
|  | ||||
|         def fake_show_decrypt(self, context, id): | ||||
|             return {'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1, | ||||
|                     'type': 'machine', 'image_state': 'decrypting'}} | ||||
|  | ||||
|         self.stubs.UnsetAll() | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show_decrypt) | ||||
|         self.assertRaises(exception.ApiError, run_instances, | ||||
|                           self.context, **kwargs) | ||||
|  | ||||
|     def test_run_instances_image_status_active(self): | ||||
|         kwargs = {'image_id': FLAGS.default_image, | ||||
|                   'instance_type': FLAGS.default_instance_type, | ||||
|                   'max_count': 1} | ||||
|         run_instances = self.cloud.run_instances | ||||
|  | ||||
|         def fake_show_stat_active(self, context, id): | ||||
|             return {'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1, | ||||
|                     'type': 'machine'}, 'status': 'active'} | ||||
|  | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show_stat_active) | ||||
|  | ||||
|         result = run_instances(self.context, **kwargs) | ||||
|         self.assertEqual(len(result['instancesSet']), 1) | ||||
|  | ||||
|     def test_terminate_instances(self): | ||||
|         inst1 = db.instance_create(self.context, {'reservation_id': 'a', | ||||
|                                                   'image_id': 1, | ||||
|                                                   'image_ref': 1, | ||||
|                                                   'host': 'host1'}) | ||||
|         terminate_instances = self.cloud.terminate_instances | ||||
|         # valid instance_id | ||||
|   | ||||
| @@ -19,7 +19,6 @@ | ||||
| Tests For Compute | ||||
| """ | ||||
|  | ||||
| import datetime | ||||
| import mox | ||||
| import stubout | ||||
|  | ||||
| @@ -84,7 +83,7 @@ class ComputeTestCase(test.TestCase): | ||||
|     def _create_instance(self, params={}): | ||||
|         """Create a test instance""" | ||||
|         inst = {} | ||||
|         inst['image_id'] = 1 | ||||
|         inst['image_ref'] = 1 | ||||
|         inst['reservation_id'] = 'r-fakeres' | ||||
|         inst['launch_time'] = '10' | ||||
|         inst['user_id'] = self.user.id | ||||
| @@ -150,7 +149,7 @@ class ComputeTestCase(test.TestCase): | ||||
|         ref = self.compute_api.create( | ||||
|                 self.context, | ||||
|                 instance_type=instance_types.get_default_instance_type(), | ||||
|                 image_id=None, | ||||
|                 image_href=None, | ||||
|                 security_group=['testgroup']) | ||||
|         try: | ||||
|             self.assertEqual(len(db.security_group_get_by_instance( | ||||
| @@ -168,7 +167,7 @@ class ComputeTestCase(test.TestCase): | ||||
|         ref = self.compute_api.create( | ||||
|                 self.context, | ||||
|                 instance_type=instance_types.get_default_instance_type(), | ||||
|                 image_id=None, | ||||
|                 image_href=None, | ||||
|                 security_group=['testgroup']) | ||||
|         try: | ||||
|             db.instance_destroy(self.context, ref[0]['id']) | ||||
| @@ -184,7 +183,7 @@ class ComputeTestCase(test.TestCase): | ||||
|         ref = self.compute_api.create( | ||||
|                 self.context, | ||||
|                 instance_type=instance_types.get_default_instance_type(), | ||||
|                 image_id=None, | ||||
|                 image_href=None, | ||||
|                 security_group=['testgroup']) | ||||
|  | ||||
|         try: | ||||
| @@ -217,12 +216,12 @@ class ComputeTestCase(test.TestCase): | ||||
|         instance_ref = db.instance_get(self.context, instance_id) | ||||
|         self.assertEqual(instance_ref['launched_at'], None) | ||||
|         self.assertEqual(instance_ref['deleted_at'], None) | ||||
|         launch = datetime.datetime.utcnow() | ||||
|         launch = utils.utcnow() | ||||
|         self.compute.run_instance(self.context, instance_id) | ||||
|         instance_ref = db.instance_get(self.context, instance_id) | ||||
|         self.assert_(instance_ref['launched_at'] > launch) | ||||
|         self.assertEqual(instance_ref['deleted_at'], None) | ||||
|         terminate = datetime.datetime.utcnow() | ||||
|         terminate = utils.utcnow() | ||||
|         self.compute.terminate_instance(self.context, instance_id) | ||||
|         self.context = self.context.elevated(True) | ||||
|         instance_ref = db.instance_get(self.context, instance_id) | ||||
|   | ||||
| @@ -20,8 +20,6 @@ | ||||
| Tests For Console proxy. | ||||
| """ | ||||
|  | ||||
| import datetime | ||||
|  | ||||
| from nova import context | ||||
| from nova import db | ||||
| from nova import exception | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
| """ | ||||
| Tests For Scheduler Host Filter Drivers. | ||||
| Tests For Scheduler Host Filters. | ||||
| """ | ||||
|  | ||||
| import json | ||||
| @@ -31,7 +31,7 @@ class FakeZoneManager: | ||||
|  | ||||
|  | ||||
| class HostFilterTestCase(test.TestCase): | ||||
|     """Test case for host filter drivers.""" | ||||
|     """Test case for host filters.""" | ||||
|  | ||||
|     def _host_caps(self, multiplier): | ||||
|         # Returns host capabilities in the following way: | ||||
| @@ -57,8 +57,8 @@ class HostFilterTestCase(test.TestCase): | ||||
|                 'host_name-label': 'xs-%s' % multiplier} | ||||
|  | ||||
|     def setUp(self): | ||||
|         self.old_flag = FLAGS.default_host_filter_driver | ||||
|         FLAGS.default_host_filter_driver = \ | ||||
|         self.old_flag = FLAGS.default_host_filter | ||||
|         FLAGS.default_host_filter = \ | ||||
|                             'nova.scheduler.host_filter.AllHostsFilter' | ||||
|         self.instance_type = dict(name='tiny', | ||||
|                 memory_mb=50, | ||||
| @@ -76,51 +76,52 @@ class HostFilterTestCase(test.TestCase): | ||||
|         self.zone_manager.service_states = states | ||||
|  | ||||
|     def tearDown(self): | ||||
|         FLAGS.default_host_filter_driver = self.old_flag | ||||
|         FLAGS.default_host_filter = self.old_flag | ||||
|  | ||||
|     def test_choose_driver(self): | ||||
|         # Test default driver ... | ||||
|         driver = host_filter.choose_driver() | ||||
|         self.assertEquals(driver._full_name(), | ||||
|     def test_choose_filter(self): | ||||
|         # Test default filter ... | ||||
|         hf = host_filter.choose_host_filter() | ||||
|         self.assertEquals(hf._full_name(), | ||||
|                         'nova.scheduler.host_filter.AllHostsFilter') | ||||
|         # Test valid driver ... | ||||
|         driver = host_filter.choose_driver( | ||||
|                         'nova.scheduler.host_filter.FlavorFilter') | ||||
|         self.assertEquals(driver._full_name(), | ||||
|                         'nova.scheduler.host_filter.FlavorFilter') | ||||
|         # Test invalid driver ... | ||||
|         # Test valid filter ... | ||||
|         hf = host_filter.choose_host_filter( | ||||
|                         'nova.scheduler.host_filter.InstanceTypeFilter') | ||||
|         self.assertEquals(hf._full_name(), | ||||
|                         'nova.scheduler.host_filter.InstanceTypeFilter') | ||||
|         # Test invalid filter ... | ||||
|         try: | ||||
|             host_filter.choose_driver('does not exist') | ||||
|             self.fail("Should not find driver") | ||||
|         except exception.SchedulerHostFilterDriverNotFound: | ||||
|             host_filter.choose_host_filter('does not exist') | ||||
|             self.fail("Should not find host filter.") | ||||
|         except exception.SchedulerHostFilterNotFound: | ||||
|             pass | ||||
|  | ||||
|     def test_all_host_driver(self): | ||||
|         driver = host_filter.AllHostsFilter() | ||||
|         cooked = driver.instance_type_to_filter(self.instance_type) | ||||
|         hosts = driver.filter_hosts(self.zone_manager, cooked) | ||||
|     def test_all_host_filter(self): | ||||
|         hf = host_filter.AllHostsFilter() | ||||
|         cooked = hf.instance_type_to_filter(self.instance_type) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|         self.assertEquals(10, len(hosts)) | ||||
|         for host, capabilities in hosts: | ||||
|             self.assertTrue(host.startswith('host')) | ||||
|  | ||||
|     def test_flavor_driver(self): | ||||
|         driver = host_filter.FlavorFilter() | ||||
|     def test_instance_type_filter(self): | ||||
|         hf = host_filter.InstanceTypeFilter() | ||||
|         # filter all hosts that can support 50 ram and 500 disk | ||||
|         name, cooked = driver.instance_type_to_filter(self.instance_type) | ||||
|         self.assertEquals('nova.scheduler.host_filter.FlavorFilter', name) | ||||
|         hosts = driver.filter_hosts(self.zone_manager, cooked) | ||||
|         name, cooked = hf.instance_type_to_filter(self.instance_type) | ||||
|         self.assertEquals('nova.scheduler.host_filter.InstanceTypeFilter', | ||||
|                           name) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|         self.assertEquals(6, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
|         self.assertEquals('host05', just_hosts[0]) | ||||
|         self.assertEquals('host10', just_hosts[5]) | ||||
|  | ||||
|     def test_json_driver(self): | ||||
|         driver = host_filter.JsonFilter() | ||||
|     def test_json_filter(self): | ||||
|         hf = host_filter.JsonFilter() | ||||
|         # filter all hosts that can support 50 ram and 500 disk | ||||
|         name, cooked = driver.instance_type_to_filter(self.instance_type) | ||||
|         name, cooked = hf.instance_type_to_filter(self.instance_type) | ||||
|         self.assertEquals('nova.scheduler.host_filter.JsonFilter', name) | ||||
|         hosts = driver.filter_hosts(self.zone_manager, cooked) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|         self.assertEquals(6, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
|         just_hosts.sort() | ||||
| @@ -132,15 +133,16 @@ class HostFilterTestCase(test.TestCase): | ||||
|         raw = ['or', | ||||
|                    ['and', | ||||
|                        ['<', '$compute.host_memory_free', 30], | ||||
|                        ['<', '$compute.disk_available', 300] | ||||
|                        ['<', '$compute.disk_available', 300], | ||||
|                    ], | ||||
|                    ['and', | ||||
|                        ['>', '$compute.host_memory_free', 70], | ||||
|                        ['>', '$compute.disk_available', 700] | ||||
|                    ] | ||||
|                        ['>', '$compute.disk_available', 700], | ||||
|                    ], | ||||
|               ] | ||||
|  | ||||
|         cooked = json.dumps(raw) | ||||
|         hosts = driver.filter_hosts(self.zone_manager, cooked) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|  | ||||
|         self.assertEquals(5, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
| @@ -152,7 +154,7 @@ class HostFilterTestCase(test.TestCase): | ||||
|                   ['=', '$compute.host_memory_free', 30], | ||||
|               ] | ||||
|         cooked = json.dumps(raw) | ||||
|         hosts = driver.filter_hosts(self.zone_manager, cooked) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|  | ||||
|         self.assertEquals(9, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
| @@ -162,7 +164,7 @@ class HostFilterTestCase(test.TestCase): | ||||
|  | ||||
|         raw = ['in', '$compute.host_memory_free', 20, 40, 60, 80, 100] | ||||
|         cooked = json.dumps(raw) | ||||
|         hosts = driver.filter_hosts(self.zone_manager, cooked) | ||||
|         hosts = hf.filter_hosts(self.zone_manager, cooked) | ||||
|  | ||||
|         self.assertEquals(5, len(hosts)) | ||||
|         just_hosts = [host for host, caps in hosts] | ||||
| @@ -174,35 +176,30 @@ class HostFilterTestCase(test.TestCase): | ||||
|         raw = ['unknown command', ] | ||||
|         cooked = json.dumps(raw) | ||||
|         try: | ||||
|             driver.filter_hosts(self.zone_manager, cooked) | ||||
|             hf.filter_hosts(self.zone_manager, cooked) | ||||
|             self.fail("Should give KeyError") | ||||
|         except KeyError, e: | ||||
|             pass | ||||
|  | ||||
|         self.assertTrue(driver.filter_hosts(self.zone_manager, json.dumps([]))) | ||||
|         self.assertTrue(driver.filter_hosts(self.zone_manager, json.dumps({}))) | ||||
|         self.assertTrue(driver.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 ['not', True, False, True, False] | ||||
|             ))) | ||||
|         self.assertTrue(hf.filter_hosts(self.zone_manager, json.dumps([]))) | ||||
|         self.assertTrue(hf.filter_hosts(self.zone_manager, json.dumps({}))) | ||||
|         self.assertTrue(hf.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 ['not', True, False, True, False]))) | ||||
|  | ||||
|         try: | ||||
|             driver.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 'not', True, False, True, False | ||||
|             )) | ||||
|             hf.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 'not', True, False, True, False)) | ||||
|             self.fail("Should give KeyError") | ||||
|         except KeyError, e: | ||||
|             pass | ||||
|  | ||||
|         self.assertFalse(driver.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 ['=', '$foo', 100] | ||||
|             ))) | ||||
|         self.assertFalse(driver.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 ['=', '$.....', 100] | ||||
|             ))) | ||||
|         self.assertFalse(driver.filter_hosts(self.zone_manager, json.dumps( | ||||
|             ['>', ['and', ['or', ['not', ['<', ['>=', ['<=', ['in', ]]]]]]]] | ||||
|         ))) | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps(['=', '$foo', 100]))) | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps(['=', '$.....', 100]))) | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps( | ||||
|             ['>', ['and', ['or', ['not', ['<', ['>=', ['<=', ['in', ]]]]]]]]))) | ||||
|  | ||||
|         self.assertFalse(driver.filter_hosts(self.zone_manager, json.dumps( | ||||
|                 ['=', {}, ['>', '$missing....foo']] | ||||
|             ))) | ||||
|         self.assertFalse(hf.filter_hosts(self.zone_manager, | ||||
|                 json.dumps(['=', {}, ['>', '$missing....foo']]))) | ||||
|   | ||||
| @@ -14,10 +14,12 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import copy | ||||
| import eventlet | ||||
| import mox | ||||
| import os | ||||
| import re | ||||
| import shutil | ||||
| import sys | ||||
|  | ||||
| from xml.etree.ElementTree import fromstring as xml_to_tree | ||||
| @@ -124,6 +126,7 @@ class CacheConcurrencyTestCase(test.TestCase): | ||||
|  | ||||
|  | ||||
| class LibvirtConnTestCase(test.TestCase): | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(LibvirtConnTestCase, self).setUp() | ||||
|         connection._late_load_cheetah() | ||||
| @@ -160,6 +163,7 @@ class LibvirtConnTestCase(test.TestCase): | ||||
|                      'vcpus':         2, | ||||
|                      'project_id':    'fake', | ||||
|                      'bridge':        'br101', | ||||
|                      'image_ref':     '123456', | ||||
|                      'instance_type_id': '5'}  # m1.small | ||||
|  | ||||
|     def lazy_load_library_exists(self): | ||||
| @@ -205,6 +209,29 @@ class LibvirtConnTestCase(test.TestCase): | ||||
|         self.mox.StubOutWithMock(connection.LibvirtConnection, '_conn') | ||||
|         connection.LibvirtConnection._conn = fake | ||||
|  | ||||
|     def fake_lookup(self, instance_name): | ||||
|  | ||||
|         class FakeVirtDomain(object): | ||||
|  | ||||
|             def snapshotCreateXML(self, *args): | ||||
|                 return None | ||||
|  | ||||
|             def XMLDesc(self, *args): | ||||
|                 return """ | ||||
|                     <domain type='kvm'> | ||||
|                         <devices> | ||||
|                             <disk type='file'> | ||||
|                                 <source file='filename'/> | ||||
|                             </disk> | ||||
|                         </devices> | ||||
|                     </domain> | ||||
|                 """ | ||||
|  | ||||
|         return FakeVirtDomain() | ||||
|  | ||||
|     def fake_execute(self, *args): | ||||
|         open(args[-1], "a").close() | ||||
|  | ||||
|     def create_service(self, **kwargs): | ||||
|         service_ref = {'host': kwargs.get('host', 'dummy'), | ||||
|                        'binary': 'nova-compute', | ||||
| @@ -280,6 +307,81 @@ class LibvirtConnTestCase(test.TestCase): | ||||
|         instance_data = dict(self.test_instance) | ||||
|         self._check_xml_and_container(instance_data) | ||||
|  | ||||
|     def test_snapshot(self): | ||||
|         if not self.lazy_load_library_exists(): | ||||
|             return | ||||
|  | ||||
|         FLAGS.image_service = 'nova.image.fake.FakeImageService' | ||||
|  | ||||
|         # Start test | ||||
|         image_service = utils.import_object(FLAGS.image_service) | ||||
|  | ||||
|         # Assuming that base image already exists in image_service | ||||
|         instance_ref = db.instance_create(self.context, self.test_instance) | ||||
|         properties = {'instance_id': instance_ref['id'], | ||||
|                       'user_id': str(self.context.user_id)} | ||||
|         snapshot_name = 'test-snap' | ||||
|         sent_meta = {'name': snapshot_name, 'is_public': False, | ||||
|                      'status': 'creating', 'properties': properties} | ||||
|         # Create new image. It will be updated in snapshot method | ||||
|         # To work with it from snapshot, the single image_service is needed | ||||
|         recv_meta = image_service.create(context, sent_meta) | ||||
|  | ||||
|         self.mox.StubOutWithMock(connection.LibvirtConnection, '_conn') | ||||
|         connection.LibvirtConnection._conn.lookupByName = self.fake_lookup | ||||
|         self.mox.StubOutWithMock(connection.utils, 'execute') | ||||
|         connection.utils.execute = self.fake_execute | ||||
|  | ||||
|         self.mox.ReplayAll() | ||||
|  | ||||
|         conn = connection.LibvirtConnection(False) | ||||
|         conn.snapshot(instance_ref, recv_meta['id']) | ||||
|  | ||||
|         snapshot = image_service.show(context, recv_meta['id']) | ||||
|         self.assertEquals(snapshot['properties']['image_state'], 'available') | ||||
|         self.assertEquals(snapshot['status'], 'active') | ||||
|         self.assertEquals(snapshot['name'], snapshot_name) | ||||
|  | ||||
|     def test_snapshot_no_image_architecture(self): | ||||
|         if not self.lazy_load_library_exists(): | ||||
|             return | ||||
|  | ||||
|         FLAGS.image_service = 'nova.image.fake.FakeImageService' | ||||
|  | ||||
|         # Start test | ||||
|         image_service = utils.import_object(FLAGS.image_service) | ||||
|  | ||||
|         # Assign image_ref = 2 from nova/images/fakes for testing different | ||||
|         # base image | ||||
|         test_instance = copy.deepcopy(self.test_instance) | ||||
|         test_instance["image_ref"] = "2" | ||||
|  | ||||
|         # Assuming that base image already exists in image_service | ||||
|         instance_ref = db.instance_create(self.context, test_instance) | ||||
|         properties = {'instance_id': instance_ref['id'], | ||||
|                       'user_id': str(self.context.user_id)} | ||||
|         snapshot_name = 'test-snap' | ||||
|         sent_meta = {'name': snapshot_name, 'is_public': False, | ||||
|                      'status': 'creating', 'properties': properties} | ||||
|         # Create new image. It will be updated in snapshot method | ||||
|         # To work with it from snapshot, the single image_service is needed | ||||
|         recv_meta = image_service.create(context, sent_meta) | ||||
|  | ||||
|         self.mox.StubOutWithMock(connection.LibvirtConnection, '_conn') | ||||
|         connection.LibvirtConnection._conn.lookupByName = self.fake_lookup | ||||
|         self.mox.StubOutWithMock(connection.utils, 'execute') | ||||
|         connection.utils.execute = self.fake_execute | ||||
|  | ||||
|         self.mox.ReplayAll() | ||||
|  | ||||
|         conn = connection.LibvirtConnection(False) | ||||
|         conn.snapshot(instance_ref, recv_meta['id']) | ||||
|  | ||||
|         snapshot = image_service.show(context, recv_meta['id']) | ||||
|         self.assertEquals(snapshot['properties']['image_state'], 'available') | ||||
|         self.assertEquals(snapshot['status'], 'active') | ||||
|         self.assertEquals(snapshot['name'], snapshot_name) | ||||
|  | ||||
|     def test_multi_nic(self): | ||||
|         instance_data = dict(self.test_instance) | ||||
|         network_info = _create_network_info(2) | ||||
| @@ -645,6 +747,8 @@ class LibvirtConnTestCase(test.TestCase): | ||||
|         except Exception, e: | ||||
|             count = (0 <= str(e.message).find('Unexpected method call')) | ||||
|  | ||||
|         shutil.rmtree(os.path.join(FLAGS.instances_path, instance.name)) | ||||
|  | ||||
|         self.assertTrue(count) | ||||
|  | ||||
|     def test_get_host_ip_addr(self): | ||||
| @@ -658,6 +762,31 @@ class LibvirtConnTestCase(test.TestCase): | ||||
|         super(LibvirtConnTestCase, self).tearDown() | ||||
|  | ||||
|  | ||||
| class NWFilterFakes: | ||||
|     def __init__(self): | ||||
|         self.filters = {} | ||||
|  | ||||
|     def nwfilterLookupByName(self, name): | ||||
|         if name in self.filters: | ||||
|             return self.filters[name] | ||||
|         raise libvirt.libvirtError('Filter Not Found') | ||||
|  | ||||
|     def filterDefineXMLMock(self, xml): | ||||
|         class FakeNWFilterInternal: | ||||
|             def __init__(self, parent, name): | ||||
|                 self.name = name | ||||
|                 self.parent = parent | ||||
|  | ||||
|             def undefine(self): | ||||
|                 del self.parent.filters[self.name] | ||||
|                 pass | ||||
|         tree = xml_to_tree(xml) | ||||
|         name = tree.get('name') | ||||
|         if name not in self.filters: | ||||
|             self.filters[name] = FakeNWFilterInternal(self, name) | ||||
|         return True | ||||
|  | ||||
|  | ||||
| class IptablesFirewallTestCase(test.TestCase): | ||||
|     def setUp(self): | ||||
|         super(IptablesFirewallTestCase, self).setUp() | ||||
| @@ -675,6 +804,20 @@ class IptablesFirewallTestCase(test.TestCase): | ||||
|         self.fw = firewall.IptablesFirewallDriver( | ||||
|                       get_connection=lambda: self.fake_libvirt_connection) | ||||
|  | ||||
|     def lazy_load_library_exists(self): | ||||
|         """check if libvirt is available.""" | ||||
|         # try to connect libvirt. if fail, skip test. | ||||
|         try: | ||||
|             import libvirt | ||||
|             import libxml2 | ||||
|         except ImportError: | ||||
|             return False | ||||
|         global libvirt | ||||
|         libvirt = __import__('libvirt') | ||||
|         connection.libvirt = __import__('libvirt') | ||||
|         connection.libxml2 = __import__('libxml2') | ||||
|         return True | ||||
|  | ||||
|     def tearDown(self): | ||||
|         self.manager.delete_project(self.project) | ||||
|         self.manager.delete_user(self.user) | ||||
| @@ -880,6 +1023,40 @@ class IptablesFirewallTestCase(test.TestCase): | ||||
|         self.mox.ReplayAll() | ||||
|         self.fw.do_refresh_security_group_rules("fake") | ||||
|  | ||||
|     def test_unfilter_instance_undefines_nwfilter(self): | ||||
|         # Skip if non-libvirt environment | ||||
|         if not self.lazy_load_library_exists(): | ||||
|             return | ||||
|  | ||||
|         admin_ctxt = context.get_admin_context() | ||||
|  | ||||
|         fakefilter = NWFilterFakes() | ||||
|         self.fw.nwfilter._conn.nwfilterDefineXML =\ | ||||
|                                fakefilter.filterDefineXMLMock | ||||
|         self.fw.nwfilter._conn.nwfilterLookupByName =\ | ||||
|                                fakefilter.nwfilterLookupByName | ||||
|  | ||||
|         instance_ref = self._create_instance_ref() | ||||
|         inst_id = instance_ref['id'] | ||||
|         instance = db.instance_get(self.context, inst_id) | ||||
|  | ||||
|         ip = '10.11.12.13' | ||||
|         network_ref = db.project_get_network(self.context, 'fake') | ||||
|         fixed_ip = {'address': ip, 'network_id': network_ref['id']} | ||||
|         db.fixed_ip_create(admin_ctxt, fixed_ip) | ||||
|         db.fixed_ip_update(admin_ctxt, ip, {'allocated': True, | ||||
|                                             'instance_id': inst_id}) | ||||
|         self.fw.setup_basic_filtering(instance) | ||||
|         self.fw.prepare_instance_filter(instance) | ||||
|         self.fw.apply_instance_filter(instance) | ||||
|         original_filter_count = len(fakefilter.filters) | ||||
|         self.fw.unfilter_instance(instance) | ||||
|  | ||||
|         # should undefine just the instance filter | ||||
|         self.assertEqual(original_filter_count - len(fakefilter.filters), 1) | ||||
|  | ||||
|         db.instance_destroy(admin_ctxt, instance_ref['id']) | ||||
|  | ||||
|  | ||||
| class NWFilterTestCase(test.TestCase): | ||||
|     def setUp(self): | ||||
| @@ -1056,3 +1233,37 @@ class NWFilterTestCase(test.TestCase): | ||||
|                                                  network_info, | ||||
|                                                  "fake") | ||||
|         self.assertEquals(len(result), 3) | ||||
|  | ||||
|     def test_unfilter_instance_undefines_nwfilters(self): | ||||
|         admin_ctxt = context.get_admin_context() | ||||
|  | ||||
|         fakefilter = NWFilterFakes() | ||||
|         self.fw._conn.nwfilterDefineXML = fakefilter.filterDefineXMLMock | ||||
|         self.fw._conn.nwfilterLookupByName = fakefilter.nwfilterLookupByName | ||||
|  | ||||
|         instance_ref = self._create_instance() | ||||
|         inst_id = instance_ref['id'] | ||||
|  | ||||
|         self.security_group = self.setup_and_return_security_group() | ||||
|  | ||||
|         db.instance_add_security_group(self.context, inst_id, | ||||
|                                        self.security_group.id) | ||||
|  | ||||
|         instance = db.instance_get(self.context, inst_id) | ||||
|  | ||||
|         ip = '10.11.12.13' | ||||
|         network_ref = db.project_get_network(self.context, 'fake') | ||||
|         fixed_ip = {'address': ip, 'network_id': network_ref['id']} | ||||
|         db.fixed_ip_create(admin_ctxt, fixed_ip) | ||||
|         db.fixed_ip_update(admin_ctxt, ip, {'allocated': True, | ||||
|                                             'instance_id': inst_id}) | ||||
|         self.fw.setup_basic_filtering(instance) | ||||
|         self.fw.prepare_instance_filter(instance) | ||||
|         self.fw.apply_instance_filter(instance) | ||||
|         original_filter_count = len(fakefilter.filters) | ||||
|         self.fw.unfilter_instance(instance) | ||||
|  | ||||
|         # should undefine 2 filters: instance and instance-secgroup | ||||
|         self.assertEqual(original_filter_count - len(fakefilter.filters), 2) | ||||
|  | ||||
|         db.instance_destroy(admin_ctxt, instance_ref['id']) | ||||
|   | ||||
| @@ -16,7 +16,6 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import datetime | ||||
| import webob | ||||
| import webob.dec | ||||
| import webob.exc | ||||
|   | ||||
| @@ -21,11 +21,24 @@ import select | ||||
| from eventlet import greenpool | ||||
| from eventlet import greenthread | ||||
|  | ||||
| from nova import exception | ||||
| from nova import test | ||||
| from nova import utils | ||||
| from nova.utils import parse_mailmap, str_dict_replace | ||||
|  | ||||
|  | ||||
| class ExceptionTestCase(test.TestCase): | ||||
|     @staticmethod | ||||
|     def _raise_exc(exc): | ||||
|         raise exc() | ||||
|  | ||||
|     def test_exceptions_raise(self): | ||||
|         for name in dir(exception): | ||||
|             exc = getattr(exception, name) | ||||
|             if isinstance(exc, type): | ||||
|                 self.assertRaises(exc, self._raise_exc, exc) | ||||
|  | ||||
|  | ||||
| class ProjectTestCase(test.TestCase): | ||||
|     def test_authors_up_to_date(self): | ||||
|         topdir = os.path.normpath(os.path.dirname(__file__) + '/../../') | ||||
|   | ||||
| @@ -13,10 +13,12 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import nova | ||||
| import stubout | ||||
|  | ||||
| import nova | ||||
| from nova import context | ||||
| from nova import flags | ||||
| from nova import log | ||||
| from nova import rpc | ||||
| import nova.notifier.api | ||||
| from nova.notifier.api import notify | ||||
| @@ -24,8 +26,6 @@ from nova.notifier import no_op_notifier | ||||
| from nova.notifier import rabbit_notifier | ||||
| from nova import test | ||||
|  | ||||
| import stubout | ||||
|  | ||||
|  | ||||
| class NotifierTestCase(test.TestCase): | ||||
|     """Test case for notifications""" | ||||
| @@ -115,3 +115,22 @@ class NotifierTestCase(test.TestCase): | ||||
|         notify('publisher_id', | ||||
|                 'event_type', 'DEBUG', dict(a=3)) | ||||
|         self.assertEqual(self.test_topic, 'testnotify.debug') | ||||
|  | ||||
|     def test_error_notification(self): | ||||
|         self.stubs.Set(nova.flags.FLAGS, 'notification_driver', | ||||
|             'nova.notifier.rabbit_notifier') | ||||
|         self.stubs.Set(nova.flags.FLAGS, 'publish_errors', True) | ||||
|         LOG = log.getLogger('nova') | ||||
|         LOG.setup_from_flags() | ||||
|         msgs = [] | ||||
|  | ||||
|         def mock_cast(context, topic, data): | ||||
|             msgs.append(data) | ||||
|  | ||||
|         self.stubs.Set(nova.rpc, 'cast', mock_cast) | ||||
|         LOG.error('foo') | ||||
|         self.assertEqual(1, len(msgs)) | ||||
|         msg = msgs[0] | ||||
|         self.assertEqual(msg['event_type'], 'error_notification') | ||||
|         self.assertEqual(msg['priority'], 'ERROR') | ||||
|         self.assertEqual(msg['payload']['error'], 'foo') | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -55,8 +55,7 @@ class VMWareAPIVMTestCase(test.TestCase): | ||||
|         vmwareapi_fake.reset() | ||||
|         db_fakes.stub_out_db_instance_api(self.stubs) | ||||
|         stubs.set_stubs(self.stubs) | ||||
|         glance_stubs.stubout_glance_client(self.stubs, | ||||
|                                            glance_stubs.FakeGlance) | ||||
|         glance_stubs.stubout_glance_client(self.stubs) | ||||
|         self.conn = vmwareapi_conn.get_connection(False) | ||||
|  | ||||
|     def _create_instance_in_the_db(self): | ||||
| @@ -64,13 +63,13 @@ class VMWareAPIVMTestCase(test.TestCase): | ||||
|                   'id': 1, | ||||
|                   'project_id': self.project.id, | ||||
|                   'user_id': self.user.id, | ||||
|                   'image_id': "1", | ||||
|                   'image_ref': "1", | ||||
|                   'kernel_id': "1", | ||||
|                   'ramdisk_id': "1", | ||||
|                   'instance_type': 'm1.large', | ||||
|                   'mac_address': 'aa:bb:cc:dd:ee:ff', | ||||
|                   } | ||||
|         self.instance = db.instance_create(values) | ||||
|         self.instance = db.instance_create(None, values) | ||||
|  | ||||
|     def _create_vm(self): | ||||
|         """Create and spawn the VM.""" | ||||
|   | ||||
| @@ -78,10 +78,12 @@ class VolumeTestCase(test.TestCase): | ||||
|         self.volume.create_snapshot(self.context, volume_src_id, snapshot_id) | ||||
|         volume_dst_id = self._create_volume(0, snapshot_id) | ||||
|         self.volume.create_volume(self.context, volume_dst_id, snapshot_id) | ||||
|         self.assertEqual(volume_dst_id, db.volume_get(context.get_admin_context(), | ||||
|                          volume_dst_id).id) | ||||
|         self.assertEqual(snapshot_id, db.volume_get(context.get_admin_context(), | ||||
|                          volume_dst_id).snapshot_id) | ||||
|         self.assertEqual(volume_dst_id, db.volume_get( | ||||
|                 context.get_admin_context(), | ||||
|                 volume_dst_id).id) | ||||
|         self.assertEqual(snapshot_id, db.volume_get( | ||||
|                 context.get_admin_context(), | ||||
|                 volume_dst_id).snapshot_id) | ||||
|  | ||||
|         self.volume.delete_volume(self.context, volume_dst_id) | ||||
|         self.volume.delete_snapshot(self.context, snapshot_id) | ||||
| @@ -211,8 +213,9 @@ class VolumeTestCase(test.TestCase): | ||||
|         self.volume.create_volume(self.context, volume_id) | ||||
|         snapshot_id = self._create_snapshot(volume_id) | ||||
|         self.volume.create_snapshot(self.context, volume_id, snapshot_id) | ||||
|         self.assertEqual(snapshot_id, db.snapshot_get(context.get_admin_context(), | ||||
|                          snapshot_id).id) | ||||
|         self.assertEqual(snapshot_id, | ||||
|                          db.snapshot_get(context.get_admin_context(), | ||||
|                                          snapshot_id).id) | ||||
|  | ||||
|         self.volume.delete_snapshot(self.context, snapshot_id) | ||||
|         self.assertRaises(exception.NotFound, | ||||
|   | ||||
| @@ -79,7 +79,7 @@ class XenAPIVolumeTestCase(test.TestCase): | ||||
|         self.values = {'id': 1, | ||||
|                   'project_id': 'fake', | ||||
|                   'user_id': 'fake', | ||||
|                   'image_id': 1, | ||||
|                   'image_ref': 1, | ||||
|                   'kernel_id': 2, | ||||
|                   'ramdisk_id': 3, | ||||
|                   'instance_type_id': '3',  # m1.large | ||||
| @@ -193,8 +193,7 @@ class XenAPIVMTestCase(test.TestCase): | ||||
|         stubs.stubout_is_vdi_pv(self.stubs) | ||||
|         self.stubs.Set(VMOps, 'reset_network', reset_network) | ||||
|         stubs.stub_out_vm_methods(self.stubs) | ||||
|         glance_stubs.stubout_glance_client(self.stubs, | ||||
|                                            glance_stubs.FakeGlance) | ||||
|         glance_stubs.stubout_glance_client(self.stubs) | ||||
|         fake_utils.stub_out_utils_execute(self.stubs) | ||||
|         self.context = context.RequestContext('fake', 'fake', False) | ||||
|         self.conn = xenapi_conn.get_connection(False) | ||||
| @@ -207,7 +206,7 @@ class XenAPIVMTestCase(test.TestCase): | ||||
|                 'id': id, | ||||
|                 'project_id': proj, | ||||
|                 'user_id': user, | ||||
|                 'image_id': 1, | ||||
|                 'image_ref': 1, | ||||
|                 'kernel_id': 2, | ||||
|                 'ramdisk_id': 3, | ||||
|                 'instance_type_id': '3',  # m1.large | ||||
| @@ -332,7 +331,7 @@ class XenAPIVMTestCase(test.TestCase): | ||||
|  | ||||
|     def check_vm_params_for_linux(self): | ||||
|         self.assertEquals(self.vm['platform']['nx'], 'false') | ||||
|         self.assertEquals(self.vm['PV_args'], 'clocksource=jiffies') | ||||
|         self.assertEquals(self.vm['PV_args'], '') | ||||
|         self.assertEquals(self.vm['PV_bootloader'], 'pygrub') | ||||
|  | ||||
|         # check that these are not set | ||||
| @@ -351,14 +350,14 @@ class XenAPIVMTestCase(test.TestCase): | ||||
|         self.assertEquals(self.vm['HVM_boot_params'], {}) | ||||
|         self.assertEquals(self.vm['HVM_boot_policy'], '') | ||||
|  | ||||
|     def _test_spawn(self, image_id, kernel_id, ramdisk_id, | ||||
|     def _test_spawn(self, image_ref, kernel_id, ramdisk_id, | ||||
|                     instance_type_id="3", os_type="linux", | ||||
|                     instance_id=1, check_injection=False): | ||||
|         stubs.stubout_loopingcall_start(self.stubs) | ||||
|         values = {'id': instance_id, | ||||
|                   'project_id': self.project.id, | ||||
|                   'user_id': self.user.id, | ||||
|                   'image_id': image_id, | ||||
|                   'image_ref': image_ref, | ||||
|                   'kernel_id': kernel_id, | ||||
|                   'ramdisk_id': ramdisk_id, | ||||
|                   'instance_type_id': instance_type_id, | ||||
| @@ -567,7 +566,7 @@ class XenAPIVMTestCase(test.TestCase): | ||||
|             'id': 1, | ||||
|             'project_id': self.project.id, | ||||
|             'user_id': self.user.id, | ||||
|             'image_id': 1, | ||||
|             'image_ref': 1, | ||||
|             'kernel_id': 2, | ||||
|             'ramdisk_id': 3, | ||||
|             'instance_type_id': '3',  # m1.large | ||||
| @@ -592,11 +591,29 @@ class XenAPIDiffieHellmanTestCase(test.TestCase): | ||||
|         bob_shared = self.bob.compute_shared(alice_pub) | ||||
|         self.assertEquals(alice_shared, bob_shared) | ||||
|  | ||||
|     def test_encryption(self): | ||||
|         msg = "This is a top-secret message" | ||||
|         enc = self.alice.encrypt(msg) | ||||
|     def _test_encryption(self, message): | ||||
|         enc = self.alice.encrypt(message) | ||||
|         self.assertFalse(enc.endswith('\n')) | ||||
|         dec = self.bob.decrypt(enc) | ||||
|         self.assertEquals(dec, msg) | ||||
|         self.assertEquals(dec, message) | ||||
|  | ||||
|     def test_encrypt_simple_message(self): | ||||
|         self._test_encryption('This is a simple message.') | ||||
|  | ||||
|     def test_encrypt_message_with_newlines_at_end(self): | ||||
|         self._test_encryption('This message has a newline at the end.\n') | ||||
|  | ||||
|     def test_encrypt_many_newlines_at_end(self): | ||||
|         self._test_encryption('Message with lotsa newlines.\n\n\n') | ||||
|  | ||||
|     def test_encrypt_newlines_inside_message(self): | ||||
|         self._test_encryption('Message\nwith\ninterior\nnewlines.') | ||||
|  | ||||
|     def test_encrypt_with_leading_newlines(self): | ||||
|         self._test_encryption('\n\nMessage with leading newlines.') | ||||
|  | ||||
|     def test_encrypt_really_long_message(self): | ||||
|         self._test_encryption(''.join(['abcd' for i in xrange(1024)])) | ||||
|  | ||||
|     def tearDown(self): | ||||
|         super(XenAPIDiffieHellmanTestCase, self).tearDown() | ||||
| @@ -623,7 +640,7 @@ class XenAPIMigrateInstance(test.TestCase): | ||||
|         self.values = {'id': 1, | ||||
|                   'project_id': self.project.id, | ||||
|                   'user_id': self.user.id, | ||||
|                   'image_id': 1, | ||||
|                   'image_ref': 1, | ||||
|                   'kernel_id': None, | ||||
|                   'ramdisk_id': None, | ||||
|                   'local_gb': 5, | ||||
| @@ -634,8 +651,7 @@ class XenAPIMigrateInstance(test.TestCase): | ||||
|         fake_utils.stub_out_utils_execute(self.stubs) | ||||
|         stubs.stub_out_migration_methods(self.stubs) | ||||
|         stubs.stubout_get_this_vm_uuid(self.stubs) | ||||
|         glance_stubs.stubout_glance_client(self.stubs, | ||||
|                                            glance_stubs.FakeGlance) | ||||
|         glance_stubs.stubout_glance_client(self.stubs) | ||||
|  | ||||
|     def tearDown(self): | ||||
|         super(XenAPIMigrateInstance, self).tearDown() | ||||
| @@ -661,8 +677,7 @@ class XenAPIDetermineDiskImageTestCase(test.TestCase): | ||||
|     """Unit tests for code that detects the ImageType.""" | ||||
|     def setUp(self): | ||||
|         super(XenAPIDetermineDiskImageTestCase, self).setUp() | ||||
|         glance_stubs.stubout_glance_client(self.stubs, | ||||
|                                            glance_stubs.FakeGlance) | ||||
|         glance_stubs.stubout_glance_client(self.stubs) | ||||
|  | ||||
|         class FakeInstance(object): | ||||
|             pass | ||||
| @@ -679,7 +694,7 @@ class XenAPIDetermineDiskImageTestCase(test.TestCase): | ||||
|     def test_instance_disk(self): | ||||
|         """If a kernel is specified, the image type is DISK (aka machine).""" | ||||
|         FLAGS.xenapi_image_service = 'objectstore' | ||||
|         self.fake_instance.image_id = glance_stubs.FakeGlance.IMAGE_MACHINE | ||||
|         self.fake_instance.image_ref = glance_stubs.FakeGlance.IMAGE_MACHINE | ||||
|         self.fake_instance.kernel_id = glance_stubs.FakeGlance.IMAGE_KERNEL | ||||
|         self.assert_disk_type(vm_utils.ImageType.DISK) | ||||
|  | ||||
| @@ -689,7 +704,7 @@ class XenAPIDetermineDiskImageTestCase(test.TestCase): | ||||
|         DISK_RAW is assumed. | ||||
|         """ | ||||
|         FLAGS.xenapi_image_service = 'objectstore' | ||||
|         self.fake_instance.image_id = glance_stubs.FakeGlance.IMAGE_RAW | ||||
|         self.fake_instance.image_ref = glance_stubs.FakeGlance.IMAGE_RAW | ||||
|         self.fake_instance.kernel_id = None | ||||
|         self.assert_disk_type(vm_utils.ImageType.DISK_RAW) | ||||
|  | ||||
| @@ -699,7 +714,7 @@ class XenAPIDetermineDiskImageTestCase(test.TestCase): | ||||
|         this case will be 'raw'. | ||||
|         """ | ||||
|         FLAGS.xenapi_image_service = 'glance' | ||||
|         self.fake_instance.image_id = glance_stubs.FakeGlance.IMAGE_RAW | ||||
|         self.fake_instance.image_ref = glance_stubs.FakeGlance.IMAGE_RAW | ||||
|         self.fake_instance.kernel_id = None | ||||
|         self.assert_disk_type(vm_utils.ImageType.DISK_RAW) | ||||
|  | ||||
| @@ -709,7 +724,7 @@ class XenAPIDetermineDiskImageTestCase(test.TestCase): | ||||
|         this case will be 'vhd'. | ||||
|         """ | ||||
|         FLAGS.xenapi_image_service = 'glance' | ||||
|         self.fake_instance.image_id = glance_stubs.FakeGlance.IMAGE_VHD | ||||
|         self.fake_instance.image_ref = glance_stubs.FakeGlance.IMAGE_VHD | ||||
|         self.fake_instance.kernel_id = None | ||||
|         self.assert_disk_type(vm_utils.ImageType.DISK_VHD) | ||||
|  | ||||
|   | ||||
| @@ -1,119 +0,0 @@ | ||||
| # Copyright 2011 OpenStack LLC. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| #    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. | ||||
| """ | ||||
| Tests For Zone Aware Scheduler. | ||||
| """ | ||||
|  | ||||
| from nova import test | ||||
| from nova.scheduler import driver | ||||
| from nova.scheduler import zone_aware_scheduler | ||||
| from nova.scheduler import zone_manager | ||||
|  | ||||
|  | ||||
| class FakeZoneAwareScheduler(zone_aware_scheduler.ZoneAwareScheduler): | ||||
|     def filter_hosts(self, num, specs): | ||||
|         # NOTE(sirp): this is returning [(hostname, services)] | ||||
|         return self.zone_manager.service_states.items() | ||||
|  | ||||
|     def weigh_hosts(self, num, specs, hosts): | ||||
|         fake_weight = 99 | ||||
|         weighted = [] | ||||
|         for hostname, caps in hosts: | ||||
|             weighted.append(dict(weight=fake_weight, name=hostname)) | ||||
|         return weighted | ||||
|  | ||||
|  | ||||
| class FakeZoneManager(zone_manager.ZoneManager): | ||||
|     def __init__(self): | ||||
|         self.service_states = { | ||||
|                         'host1': { | ||||
|                             'compute': {'ram': 1000} | ||||
|                          }, | ||||
|                          'host2': { | ||||
|                             'compute': {'ram': 2000} | ||||
|                          }, | ||||
|                          'host3': { | ||||
|                             'compute': {'ram': 3000} | ||||
|                          } | ||||
|                      } | ||||
|  | ||||
|  | ||||
| class FakeEmptyZoneManager(zone_manager.ZoneManager): | ||||
|     def __init__(self): | ||||
|         self.service_states = {} | ||||
|  | ||||
|  | ||||
| def fake_empty_call_zone_method(context, method, specs): | ||||
|     return [] | ||||
|  | ||||
|  | ||||
| def fake_call_zone_method(context, method, specs): | ||||
|     return [ | ||||
|         ('zone1', [ | ||||
|             dict(weight=1, blob='AAAAAAA'), | ||||
|             dict(weight=111, blob='BBBBBBB'), | ||||
|             dict(weight=112, blob='CCCCCCC'), | ||||
|             dict(weight=113, blob='DDDDDDD'), | ||||
|         ]), | ||||
|         ('zone2', [ | ||||
|             dict(weight=120, blob='EEEEEEE'), | ||||
|             dict(weight=2, blob='FFFFFFF'), | ||||
|             dict(weight=122, blob='GGGGGGG'), | ||||
|             dict(weight=123, blob='HHHHHHH'), | ||||
|         ]), | ||||
|         ('zone3', [ | ||||
|             dict(weight=130, blob='IIIIIII'), | ||||
|             dict(weight=131, blob='JJJJJJJ'), | ||||
|             dict(weight=132, blob='KKKKKKK'), | ||||
|             dict(weight=3, blob='LLLLLLL'), | ||||
|         ]), | ||||
|     ] | ||||
|  | ||||
|  | ||||
| class ZoneAwareSchedulerTestCase(test.TestCase): | ||||
|     """Test case for Zone Aware Scheduler.""" | ||||
|  | ||||
|     def test_zone_aware_scheduler(self): | ||||
|         """ | ||||
|         Create a nested set of FakeZones, ensure that a select call returns the | ||||
|         appropriate build plan. | ||||
|         """ | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         self.stubs.Set(sched, '_call_zone_method', fake_call_zone_method) | ||||
|  | ||||
|         zm = FakeZoneManager() | ||||
|         sched.set_zone_manager(zm) | ||||
|  | ||||
|         fake_context = {} | ||||
|         build_plan = sched.select(fake_context, {}) | ||||
|  | ||||
|         self.assertEqual(15, len(build_plan)) | ||||
|  | ||||
|         hostnames = [plan_item['name'] | ||||
|                      for plan_item in build_plan if 'name' in plan_item] | ||||
|         self.assertEqual(3, len(hostnames)) | ||||
|  | ||||
|     def test_empty_zone_aware_scheduler(self): | ||||
|         """ | ||||
|         Ensure empty hosts & child_zones result in NoValidHosts exception. | ||||
|         """ | ||||
|         sched = FakeZoneAwareScheduler() | ||||
|         self.stubs.Set(sched, '_call_zone_method', fake_empty_call_zone_method) | ||||
|  | ||||
|         zm = FakeEmptyZoneManager() | ||||
|         sched.set_zone_manager(zm) | ||||
|  | ||||
|         fake_context = {} | ||||
|         self.assertRaises(driver.NoValidHost, sched.schedule, fake_context, {}) | ||||
| @@ -52,7 +52,7 @@ def stub_out_db_instance_api(stubs): | ||||
|             else: | ||||
|                 raise NotImplementedError() | ||||
|  | ||||
|     def fake_instance_create(values): | ||||
|     def fake_instance_create(context, values): | ||||
|         """Stubs out the db.instance_create method.""" | ||||
|  | ||||
|         type_data = INSTANCE_TYPES[values['instance_type']] | ||||
| @@ -61,7 +61,7 @@ def stub_out_db_instance_api(stubs): | ||||
|             'name': values['name'], | ||||
|             'id': values['id'], | ||||
|             'reservation_id': utils.generate_uid('r'), | ||||
|             'image_id': values['image_id'], | ||||
|             'image_ref': values['image_ref'], | ||||
|             'kernel_id': values['kernel_id'], | ||||
|             'ramdisk_id': values['ramdisk_id'], | ||||
|             'state_description': 'scheduling', | ||||
|   | ||||
| @@ -42,20 +42,6 @@ def stubout_instance_snapshot(stubs): | ||||
|  | ||||
|     stubs.Set(vm_utils.VMHelper, 'fetch_image', fake_fetch_image) | ||||
|  | ||||
|     def fake_wait_for_vhd_coalesce(session, instance_id, sr_ref, vdi_ref, | ||||
|                               original_parent_uuid): | ||||
|         from nova.virt.xenapi.fake import create_vdi | ||||
|         name_label = "instance-%s" % instance_id | ||||
|         #TODO: create fake SR record | ||||
|         sr_ref = "fakesr" | ||||
|         vdi_ref = create_vdi(name_label=name_label, read_only=False, | ||||
|                              sr_ref=sr_ref, sharable=False) | ||||
|         vdi_rec = session.get_xenapi().VDI.get_record(vdi_ref) | ||||
|         vdi_uuid = vdi_rec['uuid'] | ||||
|         return vdi_uuid | ||||
|  | ||||
|     stubs.Set(vm_utils.VMHelper, 'fetch_image', fake_fetch_image) | ||||
|  | ||||
|     def fake_parse_xmlrpc_value(val): | ||||
|         return val | ||||
|  | ||||
| @@ -251,10 +237,10 @@ class FakeSessionForMigrationTests(fake.SessionBase): | ||||
|     def __init__(self, uri): | ||||
|         super(FakeSessionForMigrationTests, self).__init__(uri) | ||||
|  | ||||
|     def VDI_get_by_uuid(*args): | ||||
|     def VDI_get_by_uuid(self, *args): | ||||
|         return 'hurr' | ||||
|  | ||||
|     def VDI_resize_online(*args): | ||||
|     def VDI_resize_online(self, *args): | ||||
|         pass | ||||
|  | ||||
|     def VM_start(self, _1, ref, _2, _3): | ||||
|   | ||||
| @@ -78,7 +78,7 @@ def WrapTwistedOptions(wrapped): | ||||
|             self._absorbParameters() | ||||
|             self._absorbHandlers() | ||||
|  | ||||
|             super(TwistedOptionsToFlags, self).__init__() | ||||
|             wrapped.__init__(self) | ||||
|  | ||||
|         def _absorbFlags(self): | ||||
|             twistd_flags = [] | ||||
| @@ -163,12 +163,12 @@ def WrapTwistedOptions(wrapped): | ||||
|         def parseArgs(self, *args): | ||||
|             # TODO(termie): figure out a decent way of dealing with args | ||||
|             #return | ||||
|             super(TwistedOptionsToFlags, self).parseArgs(*args) | ||||
|             wrapped.parseArgs(self, *args) | ||||
|  | ||||
|         def postOptions(self): | ||||
|             self._doHandlers() | ||||
|  | ||||
|             super(TwistedOptionsToFlags, self).postOptions() | ||||
|             wrapped.postOptions(self) | ||||
|  | ||||
|         def __getitem__(self, key): | ||||
|             key = key.replace('-', '_') | ||||
|   | ||||
							
								
								
									
										92
									
								
								run_tests.py
									
									
									
									
									
								
							
							
						
						
									
										92
									
								
								run_tests.py
									
									
									
									
									
								
							| @@ -56,9 +56,11 @@ To run a single test module: | ||||
| """ | ||||
|  | ||||
| import gettext | ||||
| import heapq | ||||
| import os | ||||
| import unittest | ||||
| import sys | ||||
| import time | ||||
|  | ||||
| gettext.install('nova', unicode=1) | ||||
|  | ||||
| @@ -183,9 +185,21 @@ class _NullColorizer(object): | ||||
|         self.stream.write(text) | ||||
|  | ||||
|  | ||||
| def get_elapsed_time_color(elapsed_time): | ||||
|     if elapsed_time > 1.0: | ||||
|         return 'red' | ||||
|     elif elapsed_time > 0.25: | ||||
|         return 'yellow' | ||||
|     else: | ||||
|         return 'green' | ||||
|  | ||||
|  | ||||
| class NovaTestResult(result.TextTestResult): | ||||
|     def __init__(self, *args, **kw): | ||||
|         self.show_elapsed = kw.pop('show_elapsed') | ||||
|         result.TextTestResult.__init__(self, *args, **kw) | ||||
|         self.num_slow_tests = 5 | ||||
|         self.slow_tests = []  # this is a fixed-sized heap | ||||
|         self._last_case = None | ||||
|         self.colorizer = None | ||||
|         # NOTE(vish): reset stdout for the terminal check | ||||
| @@ -200,25 +214,40 @@ class NovaTestResult(result.TextTestResult): | ||||
|     def getDescription(self, test): | ||||
|         return str(test) | ||||
|  | ||||
|     def _handleElapsedTime(self, test): | ||||
|         self.elapsed_time = time.time() - self.start_time | ||||
|         item = (self.elapsed_time, test) | ||||
|         # Record only the n-slowest tests using heap | ||||
|         if len(self.slow_tests) >= self.num_slow_tests: | ||||
|             heapq.heappushpop(self.slow_tests, item) | ||||
|         else: | ||||
|             heapq.heappush(self.slow_tests, item) | ||||
|  | ||||
|     def _writeElapsedTime(self, test): | ||||
|         color = get_elapsed_time_color(self.elapsed_time) | ||||
|         self.colorizer.write("  %.2f" % self.elapsed_time, color) | ||||
|  | ||||
|     def _writeResult(self, test, long_result, color, short_result, success): | ||||
|         if self.showAll: | ||||
|             self.colorizer.write(long_result, color) | ||||
|             if self.show_elapsed and success: | ||||
|                 self._writeElapsedTime(test) | ||||
|             self.stream.writeln() | ||||
|         elif self.dots: | ||||
|             self.stream.write(short_result) | ||||
|             self.stream.flush() | ||||
|  | ||||
|     # NOTE(vish): copied from unittest with edit to add color | ||||
|     def addSuccess(self, test): | ||||
|         unittest.TestResult.addSuccess(self, test) | ||||
|         if self.showAll: | ||||
|             self.colorizer.write("OK", 'green') | ||||
|             self.stream.writeln() | ||||
|         elif self.dots: | ||||
|             self.stream.write('.') | ||||
|             self.stream.flush() | ||||
|         self._handleElapsedTime(test) | ||||
|         self._writeResult(test, 'OK', 'green', '.', True) | ||||
|  | ||||
|     # NOTE(vish): copied from unittest with edit to add color | ||||
|     def addFailure(self, test, err): | ||||
|         unittest.TestResult.addFailure(self, test, err) | ||||
|         if self.showAll: | ||||
|             self.colorizer.write("FAIL", 'red') | ||||
|             self.stream.writeln() | ||||
|         elif self.dots: | ||||
|             self.stream.write('F') | ||||
|             self.stream.flush() | ||||
|         self._handleElapsedTime(test) | ||||
|         self._writeResult(test, 'FAIL', 'red', 'F', False) | ||||
|  | ||||
|     # NOTE(vish): copied from nose with edit to add color | ||||
|     def addError(self, test, err): | ||||
| @@ -226,6 +255,7 @@ class NovaTestResult(result.TextTestResult): | ||||
|         errorClasses. If the exception is a registered class, the | ||||
|         error will be added to the list for that class, not errors. | ||||
|         """ | ||||
|         self._handleElapsedTime(test) | ||||
|         stream = getattr(self, 'stream', None) | ||||
|         ec, ev, tb = err | ||||
|         try: | ||||
| @@ -252,14 +282,11 @@ class NovaTestResult(result.TextTestResult): | ||||
|         self.errors.append((test, exc_info)) | ||||
|         test.passed = False | ||||
|         if stream is not None: | ||||
|             if self.showAll: | ||||
|                 self.colorizer.write("ERROR", 'red') | ||||
|                 self.stream.writeln() | ||||
|             elif self.dots: | ||||
|                 stream.write('E') | ||||
|             self._writeResult(test, 'ERROR', 'red', 'E', False) | ||||
|  | ||||
|     def startTest(self, test): | ||||
|         unittest.TestResult.startTest(self, test) | ||||
|         self.start_time = time.time() | ||||
|         current_case = test.test.__class__.__name__ | ||||
|  | ||||
|         if self.showAll: | ||||
| @@ -273,21 +300,47 @@ class NovaTestResult(result.TextTestResult): | ||||
|  | ||||
|  | ||||
| class NovaTestRunner(core.TextTestRunner): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self.show_elapsed = kwargs.pop('show_elapsed') | ||||
|         core.TextTestRunner.__init__(self, *args, **kwargs) | ||||
|  | ||||
|     def _makeResult(self): | ||||
|         return NovaTestResult(self.stream, | ||||
|                               self.descriptions, | ||||
|                               self.verbosity, | ||||
|                               self.config) | ||||
|                               self.config, | ||||
|                               show_elapsed=self.show_elapsed) | ||||
|  | ||||
|     def _writeSlowTests(self, result_): | ||||
|         # Pare out 'fast' tests | ||||
|         slow_tests = [item for item in result_.slow_tests | ||||
|                       if get_elapsed_time_color(item[0]) != 'green'] | ||||
|         if slow_tests: | ||||
|             slow_total_time = sum(item[0] for item in slow_tests) | ||||
|             self.stream.writeln("Slowest %i tests took %.2f secs:" | ||||
|                                 % (len(slow_tests), slow_total_time)) | ||||
|             for elapsed_time, test in sorted(slow_tests, reverse=True): | ||||
|                 time_str = "%.2f" % elapsed_time | ||||
|                 self.stream.writeln("    %s %s" % (time_str.ljust(10), test)) | ||||
|  | ||||
|     def run(self, test): | ||||
|         result_ = core.TextTestRunner.run(self, test) | ||||
|         if self.show_elapsed: | ||||
|             self._writeSlowTests(result_) | ||||
|         return result_ | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     logging.setup() | ||||
|     # If any argument looks like a test name but doesn't have "nova.tests" in | ||||
|     # front of it, automatically add that so we don't have to type as much | ||||
|     show_elapsed = True | ||||
|     argv = [] | ||||
|     for x in sys.argv: | ||||
|         if x.startswith('test_'): | ||||
|             argv.append('nova.tests.%s' % x) | ||||
|         elif x.startswith('--hide-elapsed'): | ||||
|             show_elapsed = False | ||||
|         else: | ||||
|             argv.append(x) | ||||
|  | ||||
| @@ -300,5 +353,6 @@ if __name__ == '__main__': | ||||
|  | ||||
|     runner = NovaTestRunner(stream=c.stream, | ||||
|                             verbosity=c.verbosity, | ||||
|                             config=c) | ||||
|                             config=c, | ||||
|                             show_elapsed=show_elapsed) | ||||
|     sys.exit(not core.run(config=c, testRunner=runner, argv=argv)) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Isaku Yamahata
					Isaku Yamahata