Include launch_index when creating instances.

Fixes bug 934534.

The launch_index column in the database is used as the ami-launch-index value
provided by the metadata service.

If launch_index is not specified it defaults to 1. This is incorrect for the
single instance case (should be 0) and doesn't correctly handle multi-instance
starts.

This branch changes the default launch_index to 0 (to handle the single
instance case) and adds the launch_index in the schedulers for multi-instance
starts.

Change-Id: Ifc45abf4cd9f50f732ba4a4b68c0a6242a6c9710
This commit is contained in:
Mike Milner
2012-02-23 15:13:51 -04:00
parent 892cb60d92
commit ab8f17d56a
7 changed files with 111 additions and 3 deletions

View File

@@ -121,6 +121,7 @@ Matthew Hooker <matt@cloudscaling.com>
Michael Gundlach <michael.gundlach@rackspace.com> Michael Gundlach <michael.gundlach@rackspace.com>
Michael Still <mikal@stillhq.com> Michael Still <mikal@stillhq.com>
Mike Lundy <mike@pistoncloud.com> Mike Lundy <mike@pistoncloud.com>
Mike Milner <mike.milner@canonical.com>
Mike Pittaro <mikeyp@lahondaresearch.org> Mike Pittaro <mikeyp@lahondaresearch.org>
Mike Scherbakov <mihgen@gmail.com> Mike Scherbakov <mihgen@gmail.com>
Mikyung Kang <mkkang@isi.edu> Mikyung Kang <mkkang@isi.edu>

View File

@@ -66,6 +66,7 @@ class ChanceScheduler(driver.Scheduler):
instances = [] instances = []
for num in xrange(num_instances): for num in xrange(num_instances):
host = self._schedule(context, 'compute', request_spec, **kwargs) host = self._schedule(context, 'compute', request_spec, **kwargs)
request_spec['instance_properties']['launch_index'] = num
instance = self.create_instance_db_entry(context, request_spec) instance = self.create_instance_db_entry(context, request_spec)
driver.cast_to_compute_host(context, host, driver.cast_to_compute_host(context, host,
'run_instance', instance_uuid=instance['uuid'], **kwargs) 'run_instance', instance_uuid=instance['uuid'], **kwargs)

View File

@@ -84,6 +84,7 @@ class DistributedScheduler(driver.Scheduler):
break break
weighted_host = weighted_hosts.pop(0) weighted_host = weighted_hosts.pop(0)
request_spec['instance_properties']['launch_index'] = num
instance = self._provision_resource(elevated, weighted_host, instance = self._provision_resource(elevated, weighted_host,
request_spec, kwargs) request_spec, kwargs)

View File

@@ -97,6 +97,7 @@ class SimpleScheduler(chance.ChanceScheduler):
for num in xrange(num_instances): for num in xrange(num_instances):
host = self._schedule_instance(context, host = self._schedule_instance(context,
request_spec['instance_properties'], *_args, **_kwargs) request_spec['instance_properties'], *_args, **_kwargs)
request_spec['instance_properties']['launch_index'] = num
instance_ref = self.create_instance_db_entry(context, instance_ref = self.create_instance_db_entry(context,
request_spec) request_spec)
driver.cast_to_compute_host(context, host, 'run_instance', driver.cast_to_compute_host(context, host, 'run_instance',

View File

@@ -20,6 +20,8 @@ Tests For Chance Scheduler.
import random import random
import mox
from nova import context from nova import context
from nova import exception from nova import exception
from nova.scheduler import driver from nova.scheduler import driver
@@ -114,6 +116,57 @@ class ChanceSchedulerTestCase(test_scheduler.SchedulerTestCase):
expected = [instance1_encoded, instance2_encoded] expected = [instance1_encoded, instance2_encoded]
self.assertEqual(result, expected) self.assertEqual(result, expected)
def test_scheduler_includes_launch_index(self):
ctxt = "fake-context"
instance_opts = {'fake_opt1': 'meow'}
request_spec = {'num_instances': 2,
'instance_properties': instance_opts}
instance1 = {'uuid': 'fake-uuid1'}
instance2 = {'uuid': 'fake-uuid2'}
# create_instance_db_entry() usually does this, but we're
# stubbing it.
def _add_uuid(num):
"""Return a function that adds the provided uuid number."""
def _add_uuid_num(_, spec):
spec['instance_properties']['uuid'] = 'fake-uuid%d' % num
return _add_uuid_num
def _has_launch_index(expected_index):
"""Return a function that verifies the expected index."""
def _check_launch_index(value):
if 'instance_properties' in value:
if 'launch_index' in value['instance_properties']:
index = value['instance_properties']['launch_index']
if index == expected_index:
return True
return False
return _check_launch_index
self.mox.StubOutWithMock(self.driver, '_schedule')
self.mox.StubOutWithMock(self.driver, 'create_instance_db_entry')
self.mox.StubOutWithMock(driver, 'cast_to_compute_host')
self.mox.StubOutWithMock(driver, 'encode_instance')
# instance 1
self.driver._schedule(ctxt, 'compute', request_spec).AndReturn('host')
self.driver.create_instance_db_entry(
ctxt, mox.Func(_has_launch_index(0))
).WithSideEffects(_add_uuid(1)).AndReturn(instance1)
driver.cast_to_compute_host(ctxt, 'host', 'run_instance',
instance_uuid=instance1['uuid'])
driver.encode_instance(instance1).AndReturn(instance1)
# instance 2
self.driver._schedule(ctxt, 'compute', request_spec).AndReturn('host')
self.driver.create_instance_db_entry(
ctxt, mox.Func(_has_launch_index(1))
).WithSideEffects(_add_uuid(2)).AndReturn(instance2)
driver.cast_to_compute_host(ctxt, 'host', 'run_instance',
instance_uuid=instance2['uuid'])
driver.encode_instance(instance2).AndReturn(instance2)
self.mox.ReplayAll()
self.driver.schedule_run_instance(ctxt, request_spec)
def test_basic_schedule_run_instance_no_hosts(self): def test_basic_schedule_run_instance_no_hosts(self):
ctxt = context.RequestContext('fake', 'fake', False) ctxt = context.RequestContext('fake', 'fake', False)
ctxt_elevated = 'fake-context-elevated' ctxt_elevated = 'fake-context-elevated'

View File

@@ -16,21 +16,26 @@
Tests For Distributed Scheduler. Tests For Distributed Scheduler.
""" """
import mox
from nova import context from nova import context
from nova import exception from nova import exception
from nova.scheduler import least_cost from nova.scheduler import least_cost
from nova.scheduler import host_manager from nova.scheduler import host_manager
from nova.scheduler import distributed_scheduler
from nova import test from nova import test
from nova.tests.scheduler import fakes from nova.tests.scheduler import fakes, test_scheduler
def fake_filter_hosts(hosts, filter_properties): def fake_filter_hosts(hosts, filter_properties):
return list(hosts) return list(hosts)
class DistributedSchedulerTestCase(test.TestCase): class DistributedSchedulerTestCase(test_scheduler.SchedulerTestCase):
"""Test case for Distributed Scheduler.""" """Test case for Distributed Scheduler."""
driver_cls = distributed_scheduler.DistributedScheduler
def test_run_instance_no_hosts(self): def test_run_instance_no_hosts(self):
""" """
Ensure empty hosts & child_zones result in NoValidHosts exception. Ensure empty hosts & child_zones result in NoValidHosts exception.
@@ -76,6 +81,51 @@ class DistributedSchedulerTestCase(test.TestCase):
self.assertRaises(NotImplementedError, sched._schedule, fake_context, self.assertRaises(NotImplementedError, sched._schedule, fake_context,
"foo", {}) "foo", {})
def test_scheduler_includes_launch_index(self):
ctxt = "fake-context"
fake_kwargs = {'fake_kwarg1': 'fake_value1',
'fake_kwarg2': 'fake_value2'}
instance_opts = {'fake_opt1': 'meow'}
request_spec = {'num_instances': 2,
'instance_properties': instance_opts}
instance1 = {'uuid': 'fake-uuid1'}
instance2 = {'uuid': 'fake-uuid2'}
def _has_launch_index(expected_index):
"""Return a function that verifies the expected index."""
def _check_launch_index(value):
if 'instance_properties' in value:
if 'launch_index' in value['instance_properties']:
index = value['instance_properties']['launch_index']
if index == expected_index:
return True
return False
return _check_launch_index
class ContextFake(object):
def elevated(self):
return ctxt
context_fake = ContextFake()
self.mox.StubOutWithMock(self.driver, '_schedule')
self.mox.StubOutWithMock(self.driver, '_provision_resource')
self.driver._schedule(context_fake, 'compute',
request_spec, **fake_kwargs
).AndReturn(['host1', 'host2'])
# instance 1
self.driver._provision_resource(
ctxt, 'host1',
mox.Func(_has_launch_index(0)), fake_kwargs).AndReturn(instance1)
# instance 2
self.driver._provision_resource(
ctxt, 'host2',
mox.Func(_has_launch_index(1)), fake_kwargs).AndReturn(instance2)
self.mox.ReplayAll()
self.driver.schedule_run_instance(context_fake, request_spec,
**fake_kwargs)
def test_schedule_happy_day(self): def test_schedule_happy_day(self):
"""Make sure there's nothing glaringly wrong with _schedule() """Make sure there's nothing glaringly wrong with _schedule()
by doing a happy day pass through.""" by doing a happy day pass through."""

View File

@@ -80,7 +80,8 @@ def rpc_call_wrapper(context, topic, msg, do_cast=True):
scheduler = scheduler_driver.Scheduler scheduler = scheduler_driver.Scheduler
num_instances = request_spec.get('num_instances', 1) num_instances = request_spec.get('num_instances', 1)
instances = [] instances = []
for x in xrange(num_instances): for num in xrange(num_instances):
request_spec['instance_properties']['launch_index'] = num
instance = scheduler().create_instance_db_entry( instance = scheduler().create_instance_db_entry(
context, request_spec) context, request_spec)
encoded = scheduler_driver.encode_instance(instance) encoded = scheduler_driver.encode_instance(instance)