| # 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 json |
| import logging |
| |
| import yaml |
| |
| from heat_integrationtests.common import test |
| |
| |
| LOG = logging.getLogger(__name__) |
| |
| |
| class TemplateResourceTest(test.HeatIntegrationTest): |
| """Prove that we can use the registry in a nested provider.""" |
| def setUp(self): |
| super(TemplateResourceTest, self).setUp() |
| self.client = self.orchestration_client |
| |
| def test_nested_env(self): |
| main_templ = ''' |
| heat_template_version: 2013-05-23 |
| resources: |
| secret1: |
| type: My::NestedSecret |
| outputs: |
| secret-out: |
| value: { get_attr: [secret1, value] } |
| ''' |
| |
| nested_templ = ''' |
| heat_template_version: 2013-05-23 |
| resources: |
| secret2: |
| type: My::Secret |
| outputs: |
| value: |
| value: { get_attr: [secret2, value] } |
| ''' |
| |
| env_templ = ''' |
| resource_registry: |
| "My::Secret": "OS::Heat::RandomString" |
| "My::NestedSecret": nested.yaml |
| ''' |
| |
| stack_identifier = self.stack_create( |
| template=main_templ, |
| files={'nested.yaml': nested_templ}, |
| environment=env_templ) |
| self.assert_resource_is_a_stack(stack_identifier, 'secret1') |
| |
| def test_no_infinite_recursion(self): |
| """Prove that we can override a python resource. |
| |
| And use that resource within the template resource. |
| """ |
| |
| main_templ = ''' |
| heat_template_version: 2013-05-23 |
| resources: |
| secret1: |
| type: OS::Heat::RandomString |
| outputs: |
| secret-out: |
| value: { get_attr: [secret1, value] } |
| ''' |
| |
| nested_templ = ''' |
| heat_template_version: 2013-05-23 |
| resources: |
| secret2: |
| type: OS::Heat::RandomString |
| outputs: |
| value: |
| value: { get_attr: [secret2, value] } |
| ''' |
| |
| env_templ = ''' |
| resource_registry: |
| "OS::Heat::RandomString": nested.yaml |
| ''' |
| |
| stack_identifier = self.stack_create( |
| template=main_templ, |
| files={'nested.yaml': nested_templ}, |
| environment=env_templ) |
| self.assert_resource_is_a_stack(stack_identifier, 'secret1') |
| |
| |
| class NestedAttributesTest(test.HeatIntegrationTest): |
| """Prove that we can use the template resource references.""" |
| |
| main_templ = ''' |
| heat_template_version: 2014-10-16 |
| resources: |
| secret2: |
| type: My::NestedSecret |
| outputs: |
| old_way: |
| value: { get_attr: [secret2, nested_str]} |
| test_attr1: |
| value: { get_attr: [secret2, resource.secret1, value]} |
| test_attr2: |
| value: { get_attr: [secret2, resource.secret1.value]} |
| test_ref: |
| value: { get_resource: secret2 } |
| ''' |
| |
| env_templ = ''' |
| resource_registry: |
| "My::NestedSecret": nested.yaml |
| ''' |
| |
| def setUp(self): |
| super(NestedAttributesTest, self).setUp() |
| self.client = self.orchestration_client |
| |
| def test_stack_ref(self): |
| nested_templ = ''' |
| heat_template_version: 2014-10-16 |
| resources: |
| secret1: |
| type: OS::Heat::RandomString |
| ''' |
| stack_identifier = self.stack_create( |
| template=self.main_templ, |
| files={'nested.yaml': nested_templ}, |
| environment=self.env_templ) |
| self.assert_resource_is_a_stack(stack_identifier, 'secret2') |
| stack = self.client.stacks.get(stack_identifier) |
| test_ref = self._stack_output(stack, 'test_ref') |
| self.assertIn('arn:openstack:heat:', test_ref) |
| |
| def test_transparent_ref(self): |
| """With the addition of OS::stack_id we can now use the nested resource |
| more transparently. |
| """ |
| nested_templ = ''' |
| heat_template_version: 2014-10-16 |
| resources: |
| secret1: |
| type: OS::Heat::RandomString |
| outputs: |
| OS::stack_id: |
| value: {get_resource: secret1} |
| nested_str: |
| value: {get_attr: [secret1, value]} |
| ''' |
| stack_identifier = self.stack_create( |
| template=self.main_templ, |
| files={'nested.yaml': nested_templ}, |
| environment=self.env_templ) |
| self.assert_resource_is_a_stack(stack_identifier, 'secret2') |
| stack = self.client.stacks.get(stack_identifier) |
| test_ref = self._stack_output(stack, 'test_ref') |
| test_attr = self._stack_output(stack, 'old_way') |
| |
| self.assertNotIn('arn:openstack:heat', test_ref) |
| self.assertEqual(test_attr, test_ref) |
| |
| def test_nested_attributes(self): |
| nested_templ = ''' |
| heat_template_version: 2014-10-16 |
| resources: |
| secret1: |
| type: OS::Heat::RandomString |
| outputs: |
| nested_str: |
| value: {get_attr: [secret1, value]} |
| ''' |
| stack_identifier = self.stack_create( |
| template=self.main_templ, |
| files={'nested.yaml': nested_templ}, |
| environment=self.env_templ) |
| self.assert_resource_is_a_stack(stack_identifier, 'secret2') |
| stack = self.client.stacks.get(stack_identifier) |
| old_way = self._stack_output(stack, 'old_way') |
| test_attr1 = self._stack_output(stack, 'test_attr1') |
| test_attr2 = self._stack_output(stack, 'test_attr2') |
| |
| self.assertEqual(old_way, test_attr1) |
| self.assertEqual(old_way, test_attr2) |
| |
| |
| class TemplateResourceUpdateTest(test.HeatIntegrationTest): |
| """Prove that we can do template resource updates.""" |
| |
| main_template = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Resources: |
| the_nested: |
| Type: the.yaml |
| Properties: |
| one: my_name |
| |
| Outputs: |
| identifier: |
| Value: {Ref: the_nested} |
| value: |
| Value: {'Fn::GetAtt': [the_nested, the_str]} |
| ''' |
| |
| main_template_2 = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Resources: |
| the_nested: |
| Type: the.yaml |
| Properties: |
| one: updated_name |
| |
| Outputs: |
| identifier: |
| Value: {Ref: the_nested} |
| value: |
| Value: {'Fn::GetAtt': [the_nested, the_str]} |
| ''' |
| |
| initial_tmpl = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Parameters: |
| one: |
| Default: foo |
| Type: String |
| Resources: |
| NestedResource: |
| Type: OS::Heat::RandomString |
| Properties: |
| salt: {Ref: one} |
| Outputs: |
| the_str: |
| Value: {'Fn::GetAtt': [NestedResource, value]} |
| ''' |
| prop_change_tmpl = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Parameters: |
| one: |
| Default: yikes |
| Type: String |
| two: |
| Default: foo |
| Type: String |
| Resources: |
| NestedResource: |
| Type: OS::Heat::RandomString |
| Properties: |
| salt: {Ref: one} |
| Outputs: |
| the_str: |
| Value: {'Fn::GetAtt': [NestedResource, value]} |
| ''' |
| attr_change_tmpl = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Parameters: |
| one: |
| Default: foo |
| Type: String |
| Resources: |
| NestedResource: |
| Type: OS::Heat::RandomString |
| Properties: |
| salt: {Ref: one} |
| Outputs: |
| the_str: |
| Value: {'Fn::GetAtt': [NestedResource, value]} |
| something_else: |
| Value: just_a_string |
| ''' |
| content_change_tmpl = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Parameters: |
| one: |
| Default: foo |
| Type: String |
| Resources: |
| NestedResource: |
| Type: OS::Heat::RandomString |
| Properties: |
| salt: yum |
| Outputs: |
| the_str: |
| Value: {'Fn::GetAtt': [NestedResource, value]} |
| ''' |
| |
| EXPECTED = (UPDATE, NOCHANGE) = ('update', 'nochange') |
| scenarios = [ |
| ('no_changes', dict(template=main_template, |
| provider=initial_tmpl, |
| expect=NOCHANGE)), |
| ('main_tmpl_change', dict(template=main_template_2, |
| provider=initial_tmpl, |
| expect=UPDATE)), |
| ('provider_change', dict(template=main_template, |
| provider=content_change_tmpl, |
| expect=UPDATE)), |
| ('provider_props_change', dict(template=main_template, |
| provider=prop_change_tmpl, |
| expect=NOCHANGE)), |
| ('provider_attr_change', dict(template=main_template, |
| provider=attr_change_tmpl, |
| expect=NOCHANGE)), |
| ] |
| |
| def setUp(self): |
| super(TemplateResourceUpdateTest, self).setUp() |
| self.client = self.orchestration_client |
| |
| def test_template_resource_update_template_schema(self): |
| stack_identifier = self.stack_create( |
| template=self.main_template, |
| files={'the.yaml': self.initial_tmpl}) |
| stack = self.client.stacks.get(stack_identifier) |
| initial_id = self._stack_output(stack, 'identifier') |
| initial_val = self._stack_output(stack, 'value') |
| |
| self.update_stack(stack_identifier, |
| self.template, |
| files={'the.yaml': self.provider}) |
| stack = self.client.stacks.get(stack_identifier) |
| self.assertEqual(initial_id, |
| self._stack_output(stack, 'identifier')) |
| if self.expect == self.NOCHANGE: |
| self.assertEqual(initial_val, |
| self._stack_output(stack, 'value')) |
| else: |
| self.assertNotEqual(initial_val, |
| self._stack_output(stack, 'value')) |
| |
| |
| class TemplateResourceUpdateFailedTest(test.HeatIntegrationTest): |
| """Prove that we can do updates on a nested stack to fix a stack.""" |
| main_template = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Resources: |
| keypair: |
| Type: OS::Nova::KeyPair |
| Properties: |
| name: replace-this |
| save_private_key: false |
| server: |
| Type: server_fail.yaml |
| DependsOn: keypair |
| ''' |
| nested_templ = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Resources: |
| RealRandom: |
| Type: OS::Heat::RandomString |
| ''' |
| |
| def setUp(self): |
| super(TemplateResourceUpdateFailedTest, self).setUp() |
| self.client = self.orchestration_client |
| if self.conf.keypair_name: |
| self.keypair_name = self.conf.keypair_name |
| else: |
| self.keypair = self.create_keypair() |
| self.keypair_name = self.keypair.id |
| |
| def test_update_on_failed_create(self): |
| # create a stack with "server" dependant on "keypair", but |
| # keypair fails, so "server" is not created properly. |
| # We then fix the template and it should succeed. |
| broken_templ = self.main_template.replace('replace-this', |
| self.keypair_name) |
| stack_identifier = self.stack_create( |
| template=broken_templ, |
| files={'server_fail.yaml': self.nested_templ}, |
| expected_status='CREATE_FAILED') |
| |
| fixed_templ = self.main_template.replace('replace-this', |
| test.rand_name()) |
| self.update_stack(stack_identifier, |
| fixed_templ, |
| files={'server_fail.yaml': self.nested_templ}) |
| |
| |
| class TemplateResourceAdoptTest(test.HeatIntegrationTest): |
| """Prove that we can do template resource adopt/abandon.""" |
| |
| main_template = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Resources: |
| the_nested: |
| Type: the.yaml |
| Properties: |
| one: my_name |
| Outputs: |
| identifier: |
| Value: {Ref: the_nested} |
| value: |
| Value: {'Fn::GetAtt': [the_nested, the_str]} |
| ''' |
| |
| nested_templ = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Parameters: |
| one: |
| Default: foo |
| Type: String |
| Resources: |
| RealRandom: |
| Type: OS::Heat::RandomString |
| Properties: |
| salt: {Ref: one} |
| Outputs: |
| the_str: |
| Value: {'Fn::GetAtt': [RealRandom, value]} |
| ''' |
| |
| def setUp(self): |
| super(TemplateResourceAdoptTest, self).setUp() |
| self.client = self.orchestration_client |
| |
| def _yaml_to_json(self, yaml_templ): |
| return yaml.load(yaml_templ) |
| |
| def test_abandon(self): |
| stack_name = self._stack_rand_name() |
| self.client.stacks.create( |
| stack_name=stack_name, |
| template=self.main_template, |
| files={'the.yaml': self.nested_templ}, |
| disable_rollback=True, |
| ) |
| stack = self.client.stacks.get(stack_name) |
| stack_identifier = '%s/%s' % (stack_name, stack.id) |
| self._wait_for_stack_status(stack_identifier, 'CREATE_COMPLETE') |
| |
| info = self.client.stacks.abandon(stack_id=stack_identifier) |
| self.assertEqual(self._yaml_to_json(self.main_template), |
| info['template']) |
| self.assertEqual(self._yaml_to_json(self.nested_templ), |
| info['resources']['the_nested']['template']) |
| |
| def test_adopt(self): |
| data = { |
| 'resources': { |
| 'the_nested': { |
| "type": "the.yaml", |
| "resources": { |
| "RealRandom": { |
| "type": "OS::Heat::RandomString", |
| 'resource_data': {'value': 'goopie'}, |
| 'resource_id': 'froggy' |
| } |
| } |
| } |
| }, |
| "environment": {"parameters": {}}, |
| "template": yaml.load(self.main_template) |
| } |
| |
| stack_identifier = self.stack_adopt( |
| adopt_data=json.dumps(data), |
| files={'the.yaml': self.nested_templ}) |
| |
| self.assert_resource_is_a_stack(stack_identifier, 'the_nested') |
| stack = self.client.stacks.get(stack_identifier) |
| self.assertEqual('goopie', self._stack_output(stack, 'value')) |
| |
| |
| class TemplateResourceCheckTest(test.HeatIntegrationTest): |
| """Prove that we can do template resource check.""" |
| |
| main_template = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Resources: |
| the_nested: |
| Type: the.yaml |
| Properties: |
| one: my_name |
| Outputs: |
| identifier: |
| Value: {Ref: the_nested} |
| value: |
| Value: {'Fn::GetAtt': [the_nested, the_str]} |
| ''' |
| |
| nested_templ = ''' |
| HeatTemplateFormatVersion: '2012-12-12' |
| Parameters: |
| one: |
| Default: foo |
| Type: String |
| Resources: |
| RealRandom: |
| Type: OS::Heat::RandomString |
| Properties: |
| salt: {Ref: one} |
| Outputs: |
| the_str: |
| Value: {'Fn::GetAtt': [RealRandom, value]} |
| ''' |
| |
| def setUp(self): |
| super(TemplateResourceCheckTest, self).setUp() |
| self.client = self.orchestration_client |
| |
| def test_check(self): |
| stack_name = self._stack_rand_name() |
| self.client.stacks.create( |
| stack_name=stack_name, |
| template=self.main_template, |
| files={'the.yaml': self.nested_templ}, |
| disable_rollback=True, |
| ) |
| stack = self.client.stacks.get(stack_name) |
| stack_identifier = '%s/%s' % (stack_name, stack.id) |
| self._wait_for_stack_status(stack_identifier, 'CREATE_COMPLETE') |
| |
| self.client.actions.check(stack_id=stack_identifier) |
| self._wait_for_stack_status(stack_identifier, 'CHECK_COMPLETE') |