Fixes bug 921789 distributed_scheduler is trying to pass filter_properties to compute manager but it's not serializable.. and we shouldn't pass it anyway. Change-Id: I17ce04b71dfa3cef77a6883ce3cd6fede95a538a
		
			
				
	
	
		
			315 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			315 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# 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 Distributed Scheduler.
 | 
						|
"""
 | 
						|
 | 
						|
import json
 | 
						|
 | 
						|
from nova import context
 | 
						|
from nova import db
 | 
						|
from nova import exception
 | 
						|
from nova.scheduler import distributed_scheduler
 | 
						|
from nova.scheduler import least_cost
 | 
						|
from nova.scheduler import host_manager
 | 
						|
from nova import test
 | 
						|
from nova.tests.scheduler import fakes
 | 
						|
 | 
						|
 | 
						|
def fake_call_zone_method(context, method, specs, zones):
 | 
						|
    return [
 | 
						|
        (1, [
 | 
						|
            dict(weight=2, blob='AAAAAAA'),
 | 
						|
            dict(weight=4, blob='BBBBBBB'),
 | 
						|
            dict(weight=6, blob='CCCCCCC'),
 | 
						|
            dict(weight=8, blob='DDDDDDD'),
 | 
						|
        ]),
 | 
						|
        (2, [
 | 
						|
            dict(weight=10, blob='EEEEEEE'),
 | 
						|
            dict(weight=12, blob='FFFFFFF'),
 | 
						|
            dict(weight=14, blob='GGGGGGG'),
 | 
						|
            dict(weight=16, blob='HHHHHHH'),
 | 
						|
        ]),
 | 
						|
        (3, [
 | 
						|
            dict(weight=18, blob='IIIIIII'),
 | 
						|
            dict(weight=20, blob='JJJJJJJ'),
 | 
						|
            dict(weight=22, blob='KKKKKKK'),
 | 
						|
            dict(weight=24, blob='LLLLLLL'),
 | 
						|
        ]),
 | 
						|
    ]
 | 
						|
 | 
						|
 | 
						|
def fake_zone_get_all(context):
 | 
						|
    return [
 | 
						|
        dict(id=1, api_url='zone1',
 | 
						|
             username='admin', password='password',
 | 
						|
             weight_offset=0.0, weight_scale=1.0),
 | 
						|
        dict(id=2, api_url='zone2',
 | 
						|
             username='admin', password='password',
 | 
						|
             weight_offset=1000.0, weight_scale=1.0),
 | 
						|
        dict(id=3, api_url='zone3',
 | 
						|
             username='admin', password='password',
 | 
						|
             weight_offset=0.0, weight_scale=1000.0),
 | 
						|
    ]
 | 
						|
 | 
						|
 | 
						|
def fake_filter_hosts(hosts, filter_properties):
 | 
						|
    return list(hosts)
 | 
						|
 | 
						|
 | 
						|
class DistributedSchedulerTestCase(test.TestCase):
 | 
						|
    """Test case for Distributed Scheduler."""
 | 
						|
 | 
						|
    def test_adjust_child_weights(self):
 | 
						|
        """Make sure the weights returned by child zones are
 | 
						|
        properly adjusted based on the scale/offset in the zone
 | 
						|
        db entries.
 | 
						|
        """
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        child_results = fake_call_zone_method(None, None, None, None)
 | 
						|
        zones = fake_zone_get_all(None)
 | 
						|
        weighted_hosts = sched._adjust_child_weights(child_results, zones)
 | 
						|
        scaled = [130000, 131000, 132000, 3000]
 | 
						|
        for weighted_host in weighted_hosts:
 | 
						|
            w = weighted_host.weight
 | 
						|
            if weighted_host.zone == 'zone1':  # No change
 | 
						|
                self.assertTrue(w < 1000.0)
 | 
						|
            if weighted_host.zone == 'zone2':  # Offset +1000
 | 
						|
                self.assertTrue(w >= 1000.0 and w < 2000)
 | 
						|
            if weighted_host.zone == 'zone3':  # Scale x1000
 | 
						|
                self.assertEqual(scaled.pop(0), w)
 | 
						|
 | 
						|
    def test_run_instance_no_hosts(self):
 | 
						|
        """
 | 
						|
        Ensure empty hosts & child_zones result in NoValidHosts exception.
 | 
						|
        """
 | 
						|
        def _fake_empty_call_zone_method(*args, **kwargs):
 | 
						|
            return []
 | 
						|
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        self.stubs.Set(sched, '_call_zone_method',
 | 
						|
                       _fake_empty_call_zone_method)
 | 
						|
        self.stubs.Set(db, 'zone_get_all', fake_zone_get_all)
 | 
						|
 | 
						|
        fake_context = context.RequestContext('user', 'project')
 | 
						|
        request_spec = {'instance_type': {'memory_mb': 1, 'root_gb': 1,
 | 
						|
                                          'ephemeral_gb': 0},
 | 
						|
                        'instance_properties': {'project_id': 1}}
 | 
						|
        self.assertRaises(exception.NoValidHost, sched.schedule_run_instance,
 | 
						|
                          fake_context, request_spec)
 | 
						|
 | 
						|
    def test_run_instance_with_blob_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.
 | 
						|
        """
 | 
						|
        self.schedule_called = False
 | 
						|
        self.from_blob_called = False
 | 
						|
        self.locally_called = False
 | 
						|
        self.child_zone_called = False
 | 
						|
 | 
						|
        def _fake_schedule(*args, **kwargs):
 | 
						|
            self.schedule_called = True
 | 
						|
            return least_cost.WeightedHost(1, host='x')
 | 
						|
 | 
						|
        def _fake_make_weighted_host_from_blob(*args, **kwargs):
 | 
						|
            self.from_blob_called = True
 | 
						|
            return least_cost.WeightedHost(1, zone='x', blob='y')
 | 
						|
 | 
						|
        def _fake_provision_resource_locally(*args, **kwargs):
 | 
						|
            # Tests that filter_properties is stripped
 | 
						|
            self.assertNotIn('filter_properties', kwargs)
 | 
						|
            self.locally_called = True
 | 
						|
            return 1
 | 
						|
 | 
						|
        def _fake_ask_child_zone_to_create_instance(*args, **kwargs):
 | 
						|
            self.child_zone_called = True
 | 
						|
            return 2
 | 
						|
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        self.stubs.Set(sched, '_schedule', _fake_schedule)
 | 
						|
        self.stubs.Set(sched, '_make_weighted_host_from_blob',
 | 
						|
                       _fake_make_weighted_host_from_blob)
 | 
						|
        self.stubs.Set(sched, '_provision_resource_locally',
 | 
						|
                       _fake_provision_resource_locally)
 | 
						|
        self.stubs.Set(sched, '_ask_child_zone_to_create_instance',
 | 
						|
                       _fake_ask_child_zone_to_create_instance)
 | 
						|
        request_spec = {
 | 
						|
                'instance_properties': {},
 | 
						|
                'instance_type': {},
 | 
						|
                'filter_driver': 'nova.scheduler.host_filter.AllHostsFilter',
 | 
						|
                'blob': "Non-None blob data",
 | 
						|
            }
 | 
						|
 | 
						|
        fake_context = context.RequestContext('user', 'project')
 | 
						|
        instances = sched.schedule_run_instance(fake_context, request_spec,
 | 
						|
                filter_properties={})
 | 
						|
        self.assertTrue(instances)
 | 
						|
        self.assertFalse(self.schedule_called)
 | 
						|
        self.assertTrue(self.from_blob_called)
 | 
						|
        self.assertTrue(self.child_zone_called)
 | 
						|
        self.assertFalse(self.locally_called)
 | 
						|
        self.assertEquals(instances, [2])
 | 
						|
 | 
						|
    def test_run_instance_non_admin(self):
 | 
						|
        """Test creating an instance locally using run_instance, passing
 | 
						|
        a non-admin context.  DB actions should work."""
 | 
						|
        self.was_admin = False
 | 
						|
 | 
						|
        def fake_get(context, *args, **kwargs):
 | 
						|
            # make sure this is called with admin context, even though
 | 
						|
            # we're using user context below
 | 
						|
            self.was_admin = context.is_admin
 | 
						|
            return {}
 | 
						|
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        self.stubs.Set(sched.host_manager, 'get_all_host_states', fake_get)
 | 
						|
 | 
						|
        fake_context = context.RequestContext('user', 'project')
 | 
						|
 | 
						|
        request_spec = {'instance_type': {'memory_mb': 1, 'local_gb': 1},
 | 
						|
                        'instance_properties': {'project_id': 1}}
 | 
						|
        self.assertRaises(exception.NoValidHost, sched.schedule_run_instance,
 | 
						|
                          fake_context, request_spec)
 | 
						|
        self.assertTrue(self.was_admin)
 | 
						|
 | 
						|
    def test_schedule_bad_topic(self):
 | 
						|
        """Parameter checking."""
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        fake_context = context.RequestContext('user', 'project')
 | 
						|
        self.assertRaises(NotImplementedError, sched._schedule, fake_context,
 | 
						|
                          "foo", {})
 | 
						|
 | 
						|
    def test_schedule_no_instance_type(self):
 | 
						|
        """Parameter checking."""
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        request_spec = {'instance_properties': {}}
 | 
						|
        fake_context = context.RequestContext('user', 'project')
 | 
						|
        self.assertRaises(NotImplementedError, sched._schedule, fake_context,
 | 
						|
                          "compute", request_spec=request_spec)
 | 
						|
 | 
						|
    def test_schedule_happy_day(self):
 | 
						|
        """Make sure there's nothing glaringly wrong with _schedule()
 | 
						|
        by doing a happy day pass through."""
 | 
						|
 | 
						|
        self.next_weight = 1.0
 | 
						|
 | 
						|
        def _fake_weighted_sum(functions, hosts, options):
 | 
						|
            self.next_weight += 2.0
 | 
						|
            host, hostinfo = hosts[0]
 | 
						|
            return least_cost.WeightedHost(self.next_weight, host=host,
 | 
						|
                                           hostinfo=hostinfo)
 | 
						|
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        fake_context = context.RequestContext('user', 'project',
 | 
						|
                is_admin=True)
 | 
						|
 | 
						|
        self.stubs.Set(sched.host_manager, 'filter_hosts',
 | 
						|
                fake_filter_hosts)
 | 
						|
        self.stubs.Set(least_cost, 'weighted_sum', _fake_weighted_sum)
 | 
						|
        self.stubs.Set(db, 'zone_get_all', fake_zone_get_all)
 | 
						|
        self.stubs.Set(sched, '_call_zone_method', fake_call_zone_method)
 | 
						|
 | 
						|
        request_spec = {'num_instances': 10,
 | 
						|
                        'instance_type': {'memory_mb': 512, 'root_gb': 512,
 | 
						|
                                          'ephemeral_gb': 0},
 | 
						|
                        'instance_properties': {'project_id': 1}}
 | 
						|
        self.mox.ReplayAll()
 | 
						|
        weighted_hosts = sched._schedule(fake_context, 'compute',
 | 
						|
                request_spec)
 | 
						|
        self.mox.VerifyAll()
 | 
						|
        self.assertEquals(len(weighted_hosts), 10)
 | 
						|
        for weighted_host in weighted_hosts:
 | 
						|
            # We set this up so remote hosts have even weights ...
 | 
						|
            if int(weighted_host.weight) % 2 == 0:
 | 
						|
                self.assertTrue(weighted_host.zone is not None)
 | 
						|
                self.assertTrue(weighted_host.host_state is None)
 | 
						|
            else:
 | 
						|
                self.assertTrue(weighted_host.host_state is not None)
 | 
						|
                self.assertTrue(weighted_host.zone is None)
 | 
						|
 | 
						|
    def test_schedule_local_zone(self):
 | 
						|
        """Test to make sure _schedule makes no call out to zones if
 | 
						|
        local_zone_only in the filter_properties is True.
 | 
						|
        """
 | 
						|
 | 
						|
        self.next_weight = 1.0
 | 
						|
 | 
						|
        def _fake_weighted_sum(functions, hosts, options):
 | 
						|
            self.next_weight += 2.0
 | 
						|
            host = hosts[0]
 | 
						|
            return least_cost.WeightedHost(self.next_weight, host_state=host)
 | 
						|
 | 
						|
        sched = fakes.FakeDistributedScheduler()
 | 
						|
        fake_context = context.RequestContext('user', 'project',
 | 
						|
                is_admin=True)
 | 
						|
 | 
						|
        fakes.mox_host_manager_db_calls(self.mox, fake_context)
 | 
						|
 | 
						|
        self.stubs.Set(sched.host_manager, 'filter_hosts',
 | 
						|
                fake_filter_hosts)
 | 
						|
        self.stubs.Set(least_cost, 'weighted_sum', _fake_weighted_sum)
 | 
						|
        self.stubs.Set(db, 'zone_get_all', fake_zone_get_all)
 | 
						|
        self.stubs.Set(sched, '_call_zone_method', fake_call_zone_method)
 | 
						|
 | 
						|
        request_spec = {'num_instances': 10,
 | 
						|
                        'instance_type': {'memory_mb': 512, 'root_gb': 512,
 | 
						|
                                          'ephemeral_gb': 256},
 | 
						|
                        'instance_properties': {'project_id': 1,
 | 
						|
                                                'memory_mb': 512,
 | 
						|
                                                'root_gb': 512,
 | 
						|
                                                'ephemeral_gb': 0,
 | 
						|
                                                'vcpus': 1}}
 | 
						|
        filter_properties = {'local_zone_only': True}
 | 
						|
        self.mox.ReplayAll()
 | 
						|
        weighted_hosts = sched._schedule(fake_context, 'compute',
 | 
						|
                request_spec, filter_properties=filter_properties)
 | 
						|
        self.mox.VerifyAll()
 | 
						|
        self.assertEquals(len(weighted_hosts), 10)
 | 
						|
        for weighted_host in weighted_hosts:
 | 
						|
            # There should be no remote hosts
 | 
						|
            self.assertTrue(weighted_host.host_state is not None)
 | 
						|
            self.assertTrue(weighted_host.zone is None)
 | 
						|
 | 
						|
    def test_decrypt_blob(self):
 | 
						|
        """Test that the decrypt method works."""
 | 
						|
 | 
						|
        fixture = fakes.FakeDistributedScheduler()
 | 
						|
        test_data = {'weight': 1, 'host': 'x', 'blob': 'y', 'zone': 'z'}
 | 
						|
 | 
						|
        class StubDecryptor(object):
 | 
						|
            def decryptor(self, key):
 | 
						|
                return lambda blob: blob
 | 
						|
 | 
						|
        self.stubs.Set(distributed_scheduler, 'crypto', StubDecryptor())
 | 
						|
 | 
						|
        weighted_host = fixture._make_weighted_host_from_blob(
 | 
						|
                                                        json.dumps(test_data))
 | 
						|
        self.assertTrue(isinstance(weighted_host, least_cost.WeightedHost))
 | 
						|
        self.assertEqual(weighted_host.to_dict(), dict(weight=1, host='x',
 | 
						|
                         blob='y', zone='z'))
 | 
						|
 | 
						|
    def test_get_cost_functions(self):
 | 
						|
        self.flags(reserved_host_memory_mb=128)
 | 
						|
        fixture = fakes.FakeDistributedScheduler()
 | 
						|
        fns = fixture.get_cost_functions()
 | 
						|
        self.assertEquals(len(fns), 1)
 | 
						|
        weight, fn = fns[0]
 | 
						|
        self.assertEquals(weight, 1.0)
 | 
						|
        hostinfo = host_manager.HostState('host', 'compute')
 | 
						|
        hostinfo.update_from_compute_node(dict(memory_mb=1000,
 | 
						|
                local_gb=0, vcpus=1))
 | 
						|
        self.assertEquals(1000 - 128, fn(hostinfo, {}))
 |