deb-heat/heat/tests/test_heat_autoscaling_group.py
tengqm 11dc22a701 Extract group functions into a utility module.
This patch extracts group checking functions into a dedicated module so
that resource implementations are simplified.  The module could later be
used by the autoscaling engine as well.

Change-Id: I99cdd8c9e8fe377e6923ab047a9c2ef08d1defad
partial-blueprint: reorg-asg-code
partial-blueprint: as-lib
2014-12-04 10:39:31 +08:00

486 lines
19 KiB
Python

# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import copy
import datetime
import itertools
from oslo.config import cfg
from oslo.utils import timeutils
from heat.common import exception
from heat.common import grouputils
from heat.common import short_id
from heat.common import template_format
from heat.engine import resource
from heat.engine import rsrc_defn
from heat.engine import scheduler
from heat.engine import stack_resource
from heat.tests import common
from heat.tests import generic_resource
from heat.tests import utils
class AutoScalingGroupTest(common.HeatTestCase):
as_template = '''
heat_template_version: 2013-05-23
description: AutoScaling Test
resources:
my-group:
properties:
max_size: 5
min_size: 1
resource:
type: ResourceWithPropsAndAttrs
properties:
Foo: hello
type: OS::Heat::AutoScalingGroup
'''
def setUp(self):
super(AutoScalingGroupTest, self).setUp()
resource._register_class('ResourceWithPropsAndAttrs',
generic_resource.ResourceWithPropsAndAttrs)
cfg.CONF.set_default('heat_waitcondition_server_url',
'http://server.test:8000/v1/waitcondition')
self.stub_keystoneclient()
self.parsed = template_format.parse(self.as_template)
def create_stack(self, t):
stack = utils.parse_stack(t)
stack.create()
self.assertEqual((stack.CREATE, stack.COMPLETE), stack.state)
return stack
def test_scaling_delete_empty(self):
properties = self.parsed['resources']['my-group']['properties']
properties['min_size'] = 0
properties['max_size'] = 0
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(0, grouputils.get_size(rsrc))
rsrc.delete()
def test_scaling_adjust_down_empty(self):
properties = self.parsed['resources']['my-group']['properties']
properties['min_size'] = 1
properties['max_size'] = 1
rsrc = self.create_stack(self.parsed)['my-group']
resources = grouputils.get_members(rsrc)
self.assertEqual(1, len(resources))
# Reduce the min size to 0, should complete without adjusting
props = copy.copy(rsrc.properties.data)
props['min_size'] = 0
update_snippet = rsrc_defn.ResourceDefinition(rsrc.name,
rsrc.type(),
props)
scheduler.TaskRunner(rsrc.update, update_snippet)()
self.assertEqual(resources, grouputils.get_members(rsrc))
# trigger adjustment to reduce to 0, there should be no more instances
rsrc.adjust(-1)
self.assertEqual(0, grouputils.get_size(rsrc))
def test_scaling_group_suspend(self):
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(1, grouputils.get_size(rsrc))
self.assertEqual((rsrc.CREATE, rsrc.COMPLETE), rsrc.state)
scheduler.TaskRunner(rsrc.suspend)()
self.assertEqual((rsrc.SUSPEND, rsrc.COMPLETE), rsrc.state)
def test_scaling_group_resume(self):
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(1, grouputils.get_size(rsrc))
self.assertEqual((rsrc.CREATE, rsrc.COMPLETE), rsrc.state)
rsrc.state_set(rsrc.SUSPEND, rsrc.COMPLETE)
for i in rsrc.nested().values():
i.state_set(rsrc.SUSPEND, rsrc.COMPLETE)
scheduler.TaskRunner(rsrc.resume)()
self.assertEqual((rsrc.RESUME, rsrc.COMPLETE), rsrc.state)
def test_scaling_group_create_error(self):
mock_create = self.patchobject(generic_resource.ResourceWithProps,
'handle_create')
mock_create.side_effect = Exception("Creation failed!")
rsrc = utils.parse_stack(self.parsed)['my-group']
self.assertRaises(exception.ResourceFailure,
scheduler.TaskRunner(rsrc.create))
self.assertEqual((rsrc.CREATE, rsrc.FAILED), rsrc.state)
self.assertEqual(0, grouputils.get_size(rsrc))
def test_scaling_group_update_ok_maxsize(self):
properties = self.parsed['resources']['my-group']['properties']
properties['min_size'] = 1
properties['max_size'] = 3
rsrc = self.create_stack(self.parsed)['my-group']
resources = grouputils.get_members(rsrc)
self.assertEqual(1, len(resources))
# Reduce the max size to 2, should complete without adjusting
props = copy.copy(rsrc.properties.data)
props['max_size'] = 2
update_snippet = rsrc_defn.ResourceDefinition(rsrc.name,
rsrc.type(),
props)
scheduler.TaskRunner(rsrc.update, update_snippet)()
self.assertEqual(resources, grouputils.get_members(rsrc))
self.assertEqual(2, rsrc.properties['max_size'])
def test_scaling_group_update_ok_minsize(self):
properties = self.parsed['resources']['my-group']['properties']
properties['min_size'] = 1
properties['max_size'] = 3
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(1, grouputils.get_size(rsrc))
props = copy.copy(rsrc.properties.data)
props['min_size'] = 2
update_snippet = rsrc_defn.ResourceDefinition(rsrc.name,
rsrc.type(),
props)
scheduler.TaskRunner(rsrc.update, update_snippet)()
self.assertEqual(2, grouputils.get_size(rsrc))
self.assertEqual(2, rsrc.properties['min_size'])
def test_scaling_group_update_ok_desired(self):
properties = self.parsed['resources']['my-group']['properties']
properties['min_size'] = 1
properties['max_size'] = 3
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(1, grouputils.get_size(rsrc))
props = copy.copy(rsrc.properties.data)
props['desired_capacity'] = 2
update_snippet = rsrc_defn.ResourceDefinition(rsrc.name,
rsrc.type(),
props)
scheduler.TaskRunner(rsrc.update, update_snippet)()
self.assertEqual(2, grouputils.get_size(rsrc))
self.assertEqual(2, rsrc.properties['desired_capacity'])
def test_scaling_group_update_ok_desired_remove(self):
properties = self.parsed['resources']['my-group']['properties']
properties['desired_capacity'] = 2
rsrc = self.create_stack(self.parsed)['my-group']
resources = grouputils.get_members(rsrc)
self.assertEqual(2, len(resources))
props = copy.copy(rsrc.properties.data)
props.pop('desired_capacity')
update_snippet = rsrc_defn.ResourceDefinition(rsrc.name,
rsrc.type(),
props)
scheduler.TaskRunner(rsrc.update, update_snippet)()
self.assertEqual(resources, grouputils.get_members(rsrc))
self.assertIsNone(rsrc.properties['desired_capacity'])
def test_scaling_group_scale_up_failure(self):
stack = self.create_stack(self.parsed)
mock_create = self.patchobject(generic_resource.ResourceWithProps,
'handle_create')
rsrc = stack['my-group']
self.assertEqual(1, grouputils.get_size(rsrc))
mock_create.side_effect = exception.Error('Bang')
self.assertRaises(exception.Error, rsrc.adjust, 1)
self.assertEqual(1, grouputils.get_size(rsrc))
def test_scaling_group_truncate_adjustment(self):
# Create initial group, 2 instances
properties = self.parsed['resources']['my-group']['properties']
properties['desired_capacity'] = 2
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(2, grouputils.get_size(rsrc))
rsrc.adjust(4)
self.assertEqual(5, grouputils.get_size(rsrc))
rsrc.adjust(-5)
self.assertEqual(1, grouputils.get_size(rsrc))
rsrc.adjust(0)
self.assertEqual(1, grouputils.get_size(rsrc))
def _do_test_scaling_group_percent(self, decrease, lowest,
increase, create, highest):
# Create initial group, 2 instances
properties = self.parsed['resources']['my-group']['properties']
properties['desired_capacity'] = 2
rsrc = self.create_stack(self.parsed)['my-group']
self.assertEqual(2, grouputils.get_size(rsrc))
# reduce by decrease %
rsrc.adjust(decrease, 'percentage_change_in_capacity')
self.assertEqual(lowest, grouputils.get_size(rsrc))
# raise by increase %
rsrc.adjust(increase, 'percentage_change_in_capacity')
self.assertEqual(highest, grouputils.get_size(rsrc))
def test_scaling_group_percent(self):
self._do_test_scaling_group_percent(-50, 1, 200, 2, 3)
def test_scaling_group_percent_round_up(self):
self._do_test_scaling_group_percent(-33, 1, 33, 1, 2)
def test_scaling_group_percent_round_down(self):
self._do_test_scaling_group_percent(-66, 1, 225, 2, 3)
def test_min_min_size(self):
self.parsed['resources']['my-group']['properties']['min_size'] = -1
stack = utils.parse_stack(self.parsed)
self.assertRaises(exception.StackValidationFailed,
stack['my-group'].validate)
def test_min_max_size(self):
self.parsed['resources']['my-group']['properties']['max_size'] = -1
stack = utils.parse_stack(self.parsed)
self.assertRaises(exception.StackValidationFailed,
stack['my-group'].validate)
def test_output_attribute_list(self):
rsrc = self.create_stack(self.parsed)['my-group']
member_names = rsrc.nested().resources.keys()
self.assertEqual(member_names, rsrc.FnGetAtt('outputs_list', 'Bar'))
def test_output_attribute_dict(self):
rsrc = self.create_stack(self.parsed)['my-group']
member_names = rsrc.nested().resources.keys()
self.assertEqual(dict((n, n) for n in member_names),
rsrc.FnGetAtt('outputs', 'Bar'))
def test_attribute_current_size(self):
rsrc = self.create_stack(self.parsed)['my-group']
mock_instances = self.patchobject(grouputils, 'get_size')
mock_instances.return_value = 3
self.assertEqual(3, rsrc.FnGetAtt('current_size'))
def test_attribute_current_size_with_path(self):
rsrc = self.create_stack(self.parsed)['my-group']
mock_instances = self.patchobject(grouputils, 'get_size')
mock_instances.return_value = 4
self.assertEqual(4, rsrc.FnGetAtt('current_size', 'name'))
class HeatScalingGroupWithCFNScalingPolicyTest(common.HeatTestCase):
as_template = '''
heat_template_version: 2013-05-23
description: AutoScaling Test
resources:
my-group:
properties:
max_size: 5
min_size: 1
resource:
type: ResourceWithProps
properties:
Foo: hello
type: OS::Heat::AutoScalingGroup
scale-up:
type: AWS::AutoScaling::ScalingPolicy
properties:
AutoScalingGroupName: {get_resource: my-group}
ScalingAdjustment: 1
AdjustmentType: ChangeInCapacity
Cooldown: 60
'''
def setUp(self):
super(HeatScalingGroupWithCFNScalingPolicyTest, self).setUp()
resource._register_class('ResourceWithProps',
generic_resource.ResourceWithProps)
cfg.CONF.set_default('heat_waitcondition_server_url',
'http://server.test:8000/v1/waitcondition')
self.stub_keystoneclient()
self.parsed = template_format.parse(self.as_template)
def create_stack(self, t):
stack = utils.parse_stack(t)
stack.create()
self.assertEqual((stack.CREATE, stack.COMPLETE), stack.state)
return stack
def test_scale_up(self):
stack = self.create_stack(self.parsed)
scale_up = stack['scale-up']
group = stack['my-group']
self.assertEqual(1, grouputils.get_size(group))
scale_up.signal()
self.assertEqual(2, grouputils.get_size(group))
def test_no_instance_list(self):
"""
The InstanceList attribute is not inherited from
AutoScalingResourceGroup's superclasses.
"""
stack = self.create_stack(self.parsed)
group = stack['my-group']
self.assertRaises(exception.InvalidTemplateAttribute,
group.FnGetAtt, 'InstanceList')
class ScalingPolicyTest(common.HeatTestCase):
# TODO(Qiming): Add more tests to the scaling policy
as_template = '''
heat_template_version: 2013-05-23
resources:
my-policy:
type: OS::Heat::ScalingPolicy
properties:
auto_scaling_group_id: {get_resource: my-group}
adjustment_type: change_in_capacity
scaling_adjustment: 1
my-group:
type: OS::Heat::AutoScalingGroup
properties:
max_size: 5
min_size: 1
resource:
type: ResourceWithProps
properties:
Foo: hello
'''
def setUp(self):
super(ScalingPolicyTest, self).setUp()
resource._register_class('ResourceWithProps',
generic_resource.ResourceWithProps)
self.stub_keystoneclient()
self.parsed = template_format.parse(self.as_template)
def test_alarm_attribute(self):
stack = utils.parse_stack(self.parsed)
stack.create()
policy = stack['my-policy']
self.assertIn("my-policy", policy.FnGetAtt('alarm_url'))
def test_signal(self):
stack = utils.parse_stack(self.parsed)
stack.create()
self.assertEqual((stack.CREATE, stack.COMPLETE), stack.state)
policy = stack['my-policy']
group = stack['my-group']
self.assertEqual("1234", policy.FnGetRefId())
self.assertEqual(1, grouputils.get_size(group))
policy.signal()
self.assertEqual(2, grouputils.get_size(group))
def test_signal_with_cooldown(self):
self.parsed['resources']['my-policy']['properties']['cooldown'] = 60
stack = utils.parse_stack(self.parsed)
stack.create()
policy = stack['my-policy']
group = stack['my-group']
self.assertEqual(1, grouputils.get_size(group))
policy.signal()
self.assertEqual(2, grouputils.get_size(group))
policy.signal()
# The second signal shouldn't have changed it because of cooldown
self.assertEqual(2, grouputils.get_size(group))
past = timeutils.strtime(timeutils.utcnow() -
datetime.timedelta(seconds=65))
policy.metadata_set({past: 'ChangeInCapacity : 1'})
policy.signal()
self.assertEqual(3, grouputils.get_size(group))
class RollingUpdatesTest(common.HeatTestCase):
as_template = '''
heat_template_version: 2013-05-23
description: AutoScaling Test
resources:
my-group:
properties:
max_size: 5
min_size: 4
rolling_updates:
min_in_service: 2
max_batch_size: 2
pause_time: 0
resource:
type: ResourceWithProps
properties:
Foo: hello
type: OS::Heat::AutoScalingGroup
'''
def setUp(self):
super(RollingUpdatesTest, self).setUp()
utils.setup_dummy_db()
resource._register_class('ResourceWithProps',
generic_resource.ResourceWithProps)
cfg.CONF.set_default('heat_waitcondition_server_url',
'http://server.test:8000/v1/waitcondition')
self.stub_keystoneclient()
self.parsed = template_format.parse(self.as_template)
generate_id = self.patchobject(short_id, 'generate_id')
generate_id.side_effect = ('id-%d' % (i,)
for i in itertools.count()).next
def create_stack(self, t):
stack = utils.parse_stack(t)
stack.create()
self.assertEqual((stack.CREATE, stack.COMPLETE), stack.state)
return stack
def test_rolling_update(self):
stack = self.create_stack(self.parsed)
rsrc = stack['my-group']
created_order = sorted(
rsrc.nested().resources,
key=lambda name: rsrc.nested().resources[name].created_time)
batches = []
def update_with_template(tmpl, env):
# keep track of the new updates to resources _in creation order_.
definitions = tmpl.resource_definitions(stack)
templates = [definitions[name] for name in created_order]
batches.append(templates)
self.patchobject(
stack_resource.StackResource, 'update_with_template',
side_effect=update_with_template, wraps=rsrc.update_with_template)
props = copy.deepcopy(rsrc.properties.data)
props['resource']['properties']['Foo'] = 'Hi'
update_snippet = rsrc_defn.ResourceDefinition(rsrc.name,
rsrc.type(),
props)
scheduler.TaskRunner(rsrc.update, update_snippet)()
props_schema = generic_resource.ResourceWithProps.properties_schema
def get_foos(defns):
return [d.properties(props_schema)['Foo'] for d in defns]
# first batch has 2 new resources
self.assertEqual(['Hi', 'Hi', 'hello', 'hello'],
get_foos(batches[0]))
# second batch has all new resources.
self.assertEqual(['Hi', 'Hi', 'Hi', 'Hi'],
get_foos(batches[1]))