 6fbcb65561
			
		
	
	6fbcb65561
	
	
	
		
			
			This change added json_formatter for 'tags' in method _do_stack_show. Change-Id: Icdd3c90d237804926657103dcc846f48f4355fde Closes-Bug: #1536492
		
			
				
	
	
		
			1615 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1615 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright 2012 OpenStack Foundation
 | |
| # 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.
 | |
| 
 | |
| import fnmatch
 | |
| import logging
 | |
| 
 | |
| from oslo_serialization import jsonutils
 | |
| from oslo_utils import strutils
 | |
| import six
 | |
| from six.moves.urllib import request
 | |
| import time
 | |
| import yaml
 | |
| 
 | |
| from heatclient.common import deployment_utils
 | |
| from heatclient.common import event_utils
 | |
| from heatclient.common import http
 | |
| from heatclient.common import template_format
 | |
| from heatclient.common import template_utils
 | |
| from heatclient.common import utils
 | |
| 
 | |
| from heatclient.openstack.common._i18n import _
 | |
| from heatclient.openstack.common._i18n import _LE
 | |
| from heatclient.openstack.common._i18n import _LW
 | |
| 
 | |
| import heatclient.exc as exc
 | |
| 
 | |
| logger = logging.getLogger(__name__)
 | |
| 
 | |
| 
 | |
| def _authenticated_fetcher(hc):
 | |
|     """A wrapper around the heat client object to fetch a template."""
 | |
| 
 | |
|     def _do(*args, **kwargs):
 | |
|         if isinstance(hc.http_client, http.SessionClient):
 | |
|             method, url = args
 | |
|             return hc.http_client.request(url, method, **kwargs).content
 | |
|         else:
 | |
|             return hc.http_client.raw_request(*args, **kwargs).content
 | |
| 
 | |
|     return _do
 | |
| 
 | |
| 
 | |
| @utils.arg('-f', '--template-file', metavar='<FILE>',
 | |
|            help=_('Path to the template.'))
 | |
| @utils.arg('-e', '--environment-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to the environment, it can be specified '
 | |
|                   'multiple times.'),
 | |
|            action='append')
 | |
| @utils.arg('--pre-create', metavar='<RESOURCE>',
 | |
|            default=None, action='append',
 | |
|            help=_('Name of a resource to set a pre-create hook to. Resources '
 | |
|                   'in nested stacks can be set using slash as a separator: '
 | |
|                   'nested_stack/another/my_resource. You can use wildcards '
 | |
|                   'to match multiple stacks or resources: '
 | |
|                   'nested_stack/an*/*_resource. This can be specified '
 | |
|                   'multiple times'))
 | |
| @utils.arg('-u', '--template-url', metavar='<URL>',
 | |
|            help=_('URL of template.'))
 | |
| @utils.arg('-o', '--template-object', metavar='<URL>',
 | |
|            help=_('URL to retrieve template object (e.g. from swift).'))
 | |
| @utils.arg('-c', '--create-timeout', metavar='<TIMEOUT>',
 | |
|            type=int,
 | |
|            help=_('Stack creation timeout in minutes. '
 | |
|                   'DEPRECATED use %(arg)s instead.')
 | |
|            % {'arg': '--timeout'})
 | |
| @utils.arg('-t', '--timeout', metavar='<TIMEOUT>',
 | |
|            type=int,
 | |
|            help=_('Stack creation timeout in minutes.'))
 | |
| @utils.arg('-r', '--enable-rollback', default=False, action="store_true",
 | |
|            help=_('Enable rollback on create/update failure.'))
 | |
| @utils.arg('-P', '--parameters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Parameter values used to create the stack. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by a semicolon.'),
 | |
|            action='append')
 | |
| @utils.arg('-Pf', '--parameter-file', metavar='<KEY=FILE>',
 | |
|            help=_('Parameter values from file used to create the stack. '
 | |
|                   'This can be specified multiple times. Parameter value '
 | |
|                   'would be the content of the file'),
 | |
|            action='append')
 | |
| @utils.arg('--poll', metavar='SECONDS', type=int, nargs='?', const=5,
 | |
|            help=_('Poll and report events until stack completes. '
 | |
|                   'Optional poll interval in seconds can be provided as '
 | |
|                   'argument, default 5.'))
 | |
| @utils.arg('name', metavar='<STACK_NAME>',
 | |
|            help=_('Name of the stack to create.'))
 | |
| @utils.arg('--tags', metavar='<TAG1,TAG2>',
 | |
|            help=_('A list of tags to associate with the stack.'))
 | |
| def do_stack_create(hc, args):
 | |
|     '''Create the stack.'''
 | |
|     tpl_files, template = template_utils.get_template_contents(
 | |
|         args.template_file,
 | |
|         args.template_url,
 | |
|         args.template_object,
 | |
|         _authenticated_fetcher(hc))
 | |
|     env_files, env = template_utils.process_multiple_environments_and_files(
 | |
|         env_paths=args.environment_file)
 | |
| 
 | |
|     if args.create_timeout:
 | |
|         logger.warning(_LW('%(arg1)s is deprecated, '
 | |
|                            'please use %(arg2)s instead'),
 | |
|                        {
 | |
|                            'arg1': '-c/--create-timeout',
 | |
|                            'arg2': '-t/--timeout'})
 | |
| 
 | |
|     if args.pre_create:
 | |
|         template_utils.hooks_to_env(env, args.pre_create, 'pre-create')
 | |
| 
 | |
|     fields = {
 | |
|         'stack_name': args.name,
 | |
|         'disable_rollback': not(args.enable_rollback),
 | |
|         'parameters': utils.format_all_parameters(args.parameters,
 | |
|                                                   args.parameter_file,
 | |
|                                                   args.template_file,
 | |
|                                                   args.template_url),
 | |
|         'template': template,
 | |
|         'files': dict(list(tpl_files.items()) + list(env_files.items())),
 | |
|         'environment': env
 | |
|     }
 | |
| 
 | |
|     if args.tags:
 | |
|         fields['tags'] = args.tags
 | |
|     timeout = args.timeout or args.create_timeout
 | |
|     if timeout:
 | |
|         fields['timeout_mins'] = timeout
 | |
| 
 | |
|     hc.stacks.create(**fields)
 | |
|     do_stack_list(hc)
 | |
|     if args.poll is not None:
 | |
|         _poll_for_events(hc, args.name, 'CREATE', args.poll)
 | |
| 
 | |
| 
 | |
| @utils.arg('-e', '--environment-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to the environment, it can be specified '
 | |
|                   'multiple times.'),
 | |
|            action='append')
 | |
| @utils.arg('-c', '--create-timeout', metavar='<TIMEOUT>',
 | |
|            type=int,
 | |
|            help=_('Stack creation timeout in minutes. '
 | |
|                   'DEPRECATED use %(arg)s instead.')
 | |
|            % {'arg': '--timeout'})
 | |
| @utils.arg('-t', '--timeout', metavar='<TIMEOUT>',
 | |
|            type=int,
 | |
|            help=_('Stack creation timeout in minutes.'))
 | |
| @utils.arg('-a', '--adopt-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to adopt stack data file.'))
 | |
| @utils.arg('-r', '--enable-rollback', default=False, action="store_true",
 | |
|            help=_('Enable rollback on create/update failure.'))
 | |
| @utils.arg('-P', '--parameters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Parameter values used to create the stack. This can be '
 | |
|                   'specified multiple times, or once with parameters '
 | |
|                   'separated by a semicolon.'),
 | |
|            action='append')
 | |
| @utils.arg('name', metavar='<STACK_NAME>',
 | |
|            help=_('Name of the stack to adopt.'))
 | |
| def do_stack_adopt(hc, args):
 | |
|     '''Adopt a stack.'''
 | |
|     env_files, env = template_utils.process_multiple_environments_and_files(
 | |
|         env_paths=args.environment_file)
 | |
| 
 | |
|     if not args.adopt_file:
 | |
|         raise exc.CommandError(_('Need to specify %(arg)s') %
 | |
|                                {'arg': '--adopt-file'})
 | |
| 
 | |
|     adopt_url = utils.normalise_file_path_to_url(args.adopt_file)
 | |
|     adopt_data = request.urlopen(adopt_url).read()
 | |
| 
 | |
|     if args.create_timeout:
 | |
|         logger.warning(_LW('%(arg1)s is deprecated, '
 | |
|                            'please use %(arg2)s instead'),
 | |
|                        {
 | |
|                            'arg1': '-c/--create-timeout',
 | |
|                            'arg2': '-t/--timeout'})
 | |
| 
 | |
|     fields = {
 | |
|         'stack_name': args.name,
 | |
|         'disable_rollback': not(args.enable_rollback),
 | |
|         'adopt_stack_data': adopt_data,
 | |
|         'parameters': utils.format_parameters(args.parameters),
 | |
|         'files': dict(list(env_files.items())),
 | |
|         'environment': env
 | |
|     }
 | |
| 
 | |
|     timeout = args.timeout or args.create_timeout
 | |
|     if timeout:
 | |
|         fields['timeout_mins'] = timeout
 | |
| 
 | |
|     hc.stacks.create(**fields)
 | |
|     do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('-f', '--template-file', metavar='<FILE>',
 | |
|            help=_('Path to the template.'))
 | |
| @utils.arg('-e', '--environment-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to the environment, it can be specified '
 | |
|                   'multiple times.'),
 | |
|            action='append')
 | |
| @utils.arg('-u', '--template-url', metavar='<URL>',
 | |
|            help=_('URL of template.'))
 | |
| @utils.arg('-o', '--template-object', metavar='<URL>',
 | |
|            help=_('URL to retrieve template object (e.g. from swift)'))
 | |
| @utils.arg('-t', '--timeout', metavar='<TIMEOUT>', type=int,
 | |
|            help=_('Stack creation timeout in minutes. This is only used '
 | |
|                   'during validation in preview.'))
 | |
| @utils.arg('-r', '--enable-rollback', default=False, action="store_true",
 | |
|            help=_('Enable rollback on failure. This option is not used during '
 | |
|                   'preview and exists only for symmetry with %(cmd)s.')
 | |
|            % {'cmd': 'stack-create'})
 | |
| @utils.arg('-P', '--parameters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Parameter values used to preview the stack. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by semicolon.'),
 | |
|            action='append')
 | |
| @utils.arg('-Pf', '--parameter-file', metavar='<KEY=FILE>',
 | |
|            help=_('Parameter values from file used to create the stack. '
 | |
|                   'This can be specified multiple times. Parameter value '
 | |
|                   'would be the content of the file'),
 | |
|            action='append')
 | |
| @utils.arg('name', metavar='<STACK_NAME>',
 | |
|            help=_('Name of the stack to preview.'))
 | |
| @utils.arg('--tags', metavar='<TAG1,TAG2>',
 | |
|            help=_('A list of tags to associate with the stack.'))
 | |
| def do_stack_preview(hc, args):
 | |
|     '''Preview the stack.'''
 | |
|     tpl_files, template = template_utils.get_template_contents(
 | |
|         args.template_file,
 | |
|         args.template_url,
 | |
|         args.template_object,
 | |
|         _authenticated_fetcher(hc))
 | |
|     env_files, env = template_utils.process_multiple_environments_and_files(
 | |
|         env_paths=args.environment_file)
 | |
| 
 | |
|     fields = {
 | |
|         'stack_name': args.name,
 | |
|         'disable_rollback': not(args.enable_rollback),
 | |
|         'timeout_mins': args.timeout,
 | |
|         'parameters': utils.format_all_parameters(args.parameters,
 | |
|                                                   args.parameter_file,
 | |
|                                                   args.template_file,
 | |
|                                                   args.template_url),
 | |
|         'template': template,
 | |
|         'files': dict(list(tpl_files.items()) + list(env_files.items())),
 | |
|         'environment': env
 | |
|     }
 | |
| 
 | |
|     if args.tags:
 | |
|         fields['tags'] = args.tags
 | |
| 
 | |
|     stack = hc.stacks.preview(**fields)
 | |
|     formatters = {
 | |
|         'description': utils.text_wrap_formatter,
 | |
|         'template_description': utils.text_wrap_formatter,
 | |
|         'stack_status_reason': utils.text_wrap_formatter,
 | |
|         'parameters': utils.json_formatter,
 | |
|         'outputs': utils.json_formatter,
 | |
|         'resources': utils.json_formatter,
 | |
|         'links': utils.link_formatter,
 | |
|     }
 | |
|     utils.print_dict(stack.to_dict(), formatters=formatters)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>', nargs='+',
 | |
|            help=_('Name or ID of stack(s) to delete.'))
 | |
| def do_stack_delete(hc, args):
 | |
|     '''Delete the stack(s).'''
 | |
|     failure_count = 0
 | |
| 
 | |
|     for sid in args.id:
 | |
|         fields = {'stack_id': sid}
 | |
|         try:
 | |
|             hc.stacks.delete(**fields)
 | |
|         except exc.HTTPNotFound as e:
 | |
|             failure_count += 1
 | |
|             print(e)
 | |
|     if failure_count == len(args.id):
 | |
|         raise exc.CommandError(_("Unable to delete any of the specified "
 | |
|                                "stacks."))
 | |
|     do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('-O', '--output-file', metavar='<FILE>',
 | |
|            help=_('file to output abandon result. '
 | |
|                   'If the option is specified, the result will be '
 | |
|                   'output into <FILE>.'))
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to abandon.'))
 | |
| def do_stack_abandon(hc, args):
 | |
|     '''Abandon the stack.
 | |
| 
 | |
|     This will delete the record of the stack from Heat, but will not delete
 | |
|     any of the underlying resources. Prints an adoptable JSON representation
 | |
|     of the stack to stdout or a file on success.
 | |
|     '''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         stack = hc.stacks.abandon(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         result = jsonutils.dumps(stack, indent=2)
 | |
|         if args.output_file is not None:
 | |
|             try:
 | |
|                 with open(args.output_file, "w") as f:
 | |
|                     f.write(result)
 | |
|             except IOError as err:
 | |
|                 print(result)
 | |
|                 raise exc.CommandError(str(err))
 | |
|         else:
 | |
|             print(result)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to suspend.'))
 | |
| def do_action_suspend(hc, args):
 | |
|     '''Suspend the stack.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         hc.actions.suspend(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to resume.'))
 | |
| def do_action_resume(hc, args):
 | |
|     '''Resume the stack.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         hc.actions.resume(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to check.'))
 | |
| def do_action_check(hc, args):
 | |
|     '''Check that stack resources are in expected states.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         hc.actions.check(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to describe.'))
 | |
| @utils.arg('--no-resolve-outputs', action="store_true",
 | |
|            help='Do not resolve outputs of the stack.')
 | |
| def do_stack_show(hc, args):
 | |
|     '''Describe the stack.'''
 | |
|     fields = {'stack_id': args.id,
 | |
|               'resolve_outputs': not args.no_resolve_outputs}
 | |
|     _do_stack_show(hc, fields)
 | |
| 
 | |
| 
 | |
| @utils.arg('-f', '--template-file', metavar='<FILE>',
 | |
|            help=_('Path to the template.'))
 | |
| @utils.arg('-e', '--environment-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to the environment, it can be specified '
 | |
|                   'multiple times.'),
 | |
|            action='append')
 | |
| @utils.arg('--pre-update', metavar='<RESOURCE>',
 | |
|            default=None, action='append',
 | |
|            help=_('Name of a resource to set a pre-update hook to. Resources '
 | |
|                   'in nested stacks can be set using slash as a separator: '
 | |
|                   'nested_stack/another/my_resource. You can use wildcards '
 | |
|                   'to match multiple stacks or resources: '
 | |
|                   'nested_stack/an*/*_resource. This can be specified '
 | |
|                   'multiple times'))
 | |
| @utils.arg('-u', '--template-url', metavar='<URL>',
 | |
|            help=_('URL of template.'))
 | |
| @utils.arg('-o', '--template-object', metavar='<URL>',
 | |
|            help=_('URL to retrieve template object (e.g. from swift).'))
 | |
| @utils.arg('-t', '--timeout', metavar='<TIMEOUT>',
 | |
|            type=int,
 | |
|            help=_('Stack update timeout in minutes.'))
 | |
| @utils.arg('-r', '--enable-rollback', default=False, action="store_true",
 | |
|            help=_('DEPRECATED! Use %(arg)s argument instead. '
 | |
|                   'Enable rollback on stack update failure. '
 | |
|                   'NOTE: default behavior is now to use the rollback value '
 | |
|                   'of existing stack.')
 | |
|            % {'arg': '--rollback'})
 | |
| @utils.arg('--rollback', default=None, metavar='<VALUE>',
 | |
|            help=_('Set rollback on update failure. '
 | |
|                   'Values %(true)s  set rollback to enabled. '
 | |
|                   'Values %(false)s set rollback to disabled. '
 | |
|                   'Default is to use the value of existing stack to be '
 | |
|                   'updated.')
 | |
|            % {'true': strutils.TRUE_STRINGS, 'false': strutils.FALSE_STRINGS})
 | |
| @utils.arg('-y', '--dry-run', default=False, action="store_true",
 | |
|            help='Do not actually perform the stack update, but show what '
 | |
|            'would be changed')
 | |
| @utils.arg('-P', '--parameters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Parameter values used to create the stack. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by a semicolon.'),
 | |
|            action='append')
 | |
| @utils.arg('-Pf', '--parameter-file', metavar='<KEY=FILE>',
 | |
|            help=_('Parameter values from file used to create the stack. '
 | |
|                   'This can be specified multiple times. Parameter value '
 | |
|                   'would be the content of the file'),
 | |
|            action='append')
 | |
| @utils.arg('-x', '--existing', default=False, action="store_true",
 | |
|            help=_('Re-use the template, parameters and environment of the '
 | |
|                   'current stack. If the template argument is omitted then '
 | |
|                   'the existing template is used. If no %(env_arg)s is '
 | |
|                   'specified then the existing environment is used. '
 | |
|                   'Parameters specified in %(arg)s will patch over the '
 | |
|                   'existing values in the current stack. Parameters omitted '
 | |
|                   'will keep the existing values.')
 | |
|            % {'arg': '--parameters', 'env_arg': '--environment-file'})
 | |
| @utils.arg('-c', '--clear-parameter', metavar='<PARAMETER>',
 | |
|            help=_('Remove the parameters from the set of parameters of '
 | |
|                   'current stack for the %(cmd)s. The default value in the '
 | |
|                   'template will be used. This can be specified multiple '
 | |
|                   'times.')
 | |
|            % {'cmd': 'stack-update'},
 | |
|            action='append')
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to update.'))
 | |
| @utils.arg('--tags', metavar='<TAG1,TAG2>',
 | |
|            help=_('An updated list of tags to associate with the stack.'))
 | |
| def do_stack_update(hc, args):
 | |
|     '''Update the stack.'''
 | |
| 
 | |
|     tpl_files, template = template_utils.get_template_contents(
 | |
|         args.template_file,
 | |
|         args.template_url,
 | |
|         args.template_object,
 | |
|         _authenticated_fetcher(hc),
 | |
|         existing=args.existing)
 | |
| 
 | |
|     env_files, env = template_utils.process_multiple_environments_and_files(
 | |
|         env_paths=args.environment_file)
 | |
| 
 | |
|     if args.pre_update:
 | |
|         template_utils.hooks_to_env(env, args.pre_update, 'pre-update')
 | |
| 
 | |
|     fields = {
 | |
|         'stack_id': args.id,
 | |
|         'parameters': utils.format_all_parameters(args.parameters,
 | |
|                                                   args.parameter_file,
 | |
|                                                   args.template_file,
 | |
|                                                   args.template_url),
 | |
|         'existing': args.existing,
 | |
|         'template': template,
 | |
|         'files': dict(list(tpl_files.items()) + list(env_files.items())),
 | |
|         'environment': env
 | |
|     }
 | |
| 
 | |
|     if args.tags:
 | |
|         fields['tags'] = args.tags
 | |
|     if args.timeout:
 | |
|         fields['timeout_mins'] = args.timeout
 | |
|     if args.clear_parameter:
 | |
|         fields['clear_parameters'] = list(args.clear_parameter)
 | |
| 
 | |
|     if args.rollback is not None:
 | |
|         try:
 | |
|             rollback = strutils.bool_from_string(args.rollback, strict=True)
 | |
|         except ValueError as ex:
 | |
|             raise exc.CommandError(six.text_type(ex))
 | |
|         else:
 | |
|             fields['disable_rollback'] = not(rollback)
 | |
|     # TODO(pshchelo): remove the following 'else' clause after deprecation
 | |
|     # period of --enable-rollback switch and assign -r shortcut to --rollback
 | |
|     else:
 | |
|         if args.enable_rollback:
 | |
|             fields['disable_rollback'] = False
 | |
| 
 | |
|     if args.dry_run is True:
 | |
|         resource_changes = hc.stacks.preview_update(**fields)
 | |
| 
 | |
|         formatters = {
 | |
|             'resource_identity': utils.json_formatter
 | |
|         }
 | |
| 
 | |
|         fields = ['state', 'resource_name', 'resource_type',
 | |
|                   'resource_identity']
 | |
| 
 | |
|         for k in resource_changes.get("resource_changes", {}).keys():
 | |
|             for i in range(len(resource_changes["resource_changes"][k])):
 | |
|                 resource_changes["resource_changes"][k][i]['state'] = k
 | |
| 
 | |
|         utils.print_update_list(
 | |
|             sum(resource_changes["resource_changes"].values(), []),
 | |
|             fields,
 | |
|             formatters=formatters
 | |
|         )
 | |
|         return
 | |
| 
 | |
|     hc.stacks.update(**fields)
 | |
|     do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to cancel update for.'))
 | |
| def do_stack_cancel_update(hc, args):
 | |
|     '''Cancel currently running update of the stack.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         hc.actions.cancel_update(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         do_stack_list(hc)
 | |
| 
 | |
| 
 | |
| @utils.arg('-s', '--show-deleted', default=False, action="store_true",
 | |
|            help=_('Include soft-deleted stacks in the stack listing.'))
 | |
| @utils.arg('-n', '--show-nested', default=False, action="store_true",
 | |
|            help=_('Include nested stacks in the stack listing.'))
 | |
| @utils.arg('-a', '--show-hidden', default=False, action="store_true",
 | |
|            help=_('Include hidden stacks in the stack listing.'))
 | |
| @utils.arg('-f', '--filters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Filter parameters to apply on returned stacks. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by a semicolon.'),
 | |
|            action='append')
 | |
| @utils.arg('-t', '--tags', metavar='<TAG1,TAG2...>',
 | |
|            help=_('Show stacks containing these tags, combine multiple tags '
 | |
|                   'using the boolean AND expression'))
 | |
| @utils.arg('--tags-any', metavar='<TAG1,TAG2...>',
 | |
|            help=_('Show stacks containing these tags, combine multiple tags '
 | |
|                   'using the boolean OR expression'))
 | |
| @utils.arg('--not-tags', metavar='<TAG1,TAG2...>',
 | |
|            help=_('Show stacks not containing these tags, combine multiple '
 | |
|                   'tags using the boolean AND expression'))
 | |
| @utils.arg('--not-tags-any', metavar='<TAG1,TAG2...>',
 | |
|            help=_('Show stacks not containing these tags, combine multiple '
 | |
|                   'tags using the boolean OR expression'))
 | |
| @utils.arg('-l', '--limit', metavar='<LIMIT>',
 | |
|            help=_('Limit the number of stacks returned.'))
 | |
| @utils.arg('-m', '--marker', metavar='<ID>',
 | |
|            help=_('Only return stacks that appear after the given stack ID.'))
 | |
| @utils.arg('-k', '--sort-keys', metavar='<KEY1;KEY2...>',
 | |
|            help=_('List of keys for sorting the returned stacks. '
 | |
|                   'This can be specified multiple times or once with keys '
 | |
|                   'separated by semicolons. Valid sorting keys include '
 | |
|                   '"stack_name", "stack_status", "creation_time" and '
 | |
|                   '"updated_time".'),
 | |
|            action='append')
 | |
| @utils.arg('-d', '--sort-dir', metavar='[asc|desc]',
 | |
|            help=_('Sorting direction (either "asc" or "desc") for the sorting '
 | |
|                   'keys.'))
 | |
| @utils.arg('-g', '--global-tenant', action='store_true', default=False,
 | |
|            help=_('Display stacks from all tenants. Operation only authorized '
 | |
|                   'for users who match the policy in heat\'s policy.json.'))
 | |
| @utils.arg('-o', '--show-owner', action='store_true', default=False,
 | |
|            help=_('Display stack owner information. This is automatically '
 | |
|                   'enabled when using %(arg)s.') % {'arg': '--global-tenant'})
 | |
| def do_stack_list(hc, args=None):
 | |
|     '''List the user's stacks.'''
 | |
|     kwargs = {}
 | |
|     fields = ['id', 'stack_name', 'stack_status', 'creation_time',
 | |
|               'updated_time']
 | |
|     sort_keys = ['stack_name', 'stack_status', 'creation_time',
 | |
|                  'updated_time']
 | |
|     sortby_index = 3
 | |
|     if args:
 | |
|         kwargs = {'limit': args.limit,
 | |
|                   'marker': args.marker,
 | |
|                   'filters': utils.format_parameters(args.filters),
 | |
|                   'tags': args.tags,
 | |
|                   'tags_any': args.tags_any,
 | |
|                   'not_tags': args.not_tags,
 | |
|                   'not_tags_any': args.not_tags_any,
 | |
|                   'global_tenant': args.global_tenant,
 | |
|                   'show_deleted': args.show_deleted,
 | |
|                   'show_hidden': args.show_hidden}
 | |
|         if args.show_nested:
 | |
|             fields.append('parent')
 | |
|             kwargs['show_nested'] = True
 | |
| 
 | |
|         if args.sort_keys:
 | |
|             # flatten key list first
 | |
|             keys = []
 | |
|             for k in args.sort_keys:
 | |
|                 if ';' in k:
 | |
|                     keys.extend(k.split(';'))
 | |
|                 else:
 | |
|                     keys.append(k)
 | |
|             # validate key list
 | |
|             for key in keys:
 | |
|                 if key not in sort_keys:
 | |
|                     err = _("Sorting key '%(key)s' not one of the supported "
 | |
|                             "keys: %(keys)s") % {'key': key, "keys": sort_keys}
 | |
|                     raise exc.CommandError(err)
 | |
|             kwargs['sort_keys'] = keys
 | |
|             sortby_index = None
 | |
| 
 | |
|         if args.sort_dir:
 | |
|             if args.sort_dir not in ('asc', 'desc'):
 | |
|                 raise exc.CommandError(_("Sorting direction must be one of "
 | |
|                                          "'asc' and 'desc'"))
 | |
|             kwargs['sort_dir'] = args.sort_dir
 | |
| 
 | |
|         if args.global_tenant or args.show_owner:
 | |
|             fields.insert(2, 'stack_owner')
 | |
|         if args.global_tenant:
 | |
|             fields.insert(2, 'project')
 | |
| 
 | |
|     stacks = hc.stacks.list(**kwargs)
 | |
|     utils.print_list(stacks, fields, sortby_index=sortby_index)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to query.'))
 | |
| def do_output_list(hc, args):
 | |
|     """Show available outputs."""
 | |
|     try:
 | |
|         outputs = hc.stacks.output_list(args.id)
 | |
|     except exc.HTTPNotFound:
 | |
|         try:
 | |
|             outputs = hc.stacks.get(args.id).to_dict()
 | |
|         except exc.HTTPNotFound:
 | |
|             raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
| 
 | |
|     fields = ['output_key', 'description']
 | |
|     formatters = {
 | |
|         'output_key': lambda x: x['output_key'],
 | |
|         'description': lambda x: x['description'],
 | |
|     }
 | |
| 
 | |
|     utils.print_list(outputs['outputs'], fields, formatters=formatters)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to query.'))
 | |
| @utils.arg('output', metavar='<OUTPUT NAME>', nargs='?', default=None,
 | |
|            help=_('Name of an output to display.'))
 | |
| @utils.arg('-F', '--format', metavar='<FORMAT>',
 | |
|            help=_('The output value format, one of: json, raw.'),
 | |
|            default='json')
 | |
| @utils.arg('-a', '--all', default=False, action='store_true',
 | |
|            help=_('Display all stack outputs.'))
 | |
| @utils.arg('-v', '--only-value', default=False, action="store_true",
 | |
|            help=_('Returns only output value in specified format.'))
 | |
| def do_output_show(hc, args):
 | |
|     """Show a specific stack output."""
 | |
|     def resolve_output(output_key):
 | |
|         try:
 | |
|             output = hc.stacks.output_show(args.id, output_key)
 | |
|         except exc.HTTPNotFound:
 | |
|             try:
 | |
|                 output = None
 | |
|                 stack = hc.stacks.get(args.id).to_dict()
 | |
|                 for o in stack.get('outputs', []):
 | |
|                     if o['output_key'] == output_key:
 | |
|                         output = {'output': o}
 | |
|                         break
 | |
|                 if output is None:
 | |
|                     raise exc.CommandError(_('Output %(key)s not found.') % {
 | |
|                         'key': args.output})
 | |
|             except exc.HTTPNotFound:
 | |
|                 raise exc.CommandError(
 | |
|                     _('Stack %(id)s or output %(key)s not found.') % {
 | |
|                         'id': args.id,
 | |
|                         'key': args.output})
 | |
|         return output
 | |
| 
 | |
|     def show_output(output):
 | |
|         if 'output_error' in output['output']:
 | |
|             msg = _("Output error: %s") % output['output']['output_error']
 | |
|             raise exc.CommandError(msg)
 | |
|         if args.only_value:
 | |
|             if (args.format == 'json'
 | |
|                     or isinstance(output['output']['output_value'], dict)
 | |
|                     or isinstance(output['output']['output_value'], list)):
 | |
|                 print(
 | |
|                     utils.json_formatter(output['output']['output_value']))
 | |
|             else:
 | |
|                 print(output['output']['output_value'])
 | |
|         else:
 | |
|             formatters = {
 | |
|                 'output_value': (lambda x: utils.json_formatter(x)
 | |
|                                  if args.format == 'json'
 | |
|                                  else x)
 | |
|             }
 | |
|             utils.print_dict(output['output'], formatters=formatters)
 | |
| 
 | |
|     if args.all:
 | |
|         try:
 | |
|             outputs = hc.stacks.output_list(args.id)
 | |
|             resolved = False
 | |
|         except exc.HTTPNotFound:
 | |
|             try:
 | |
|                 outputs = hc.stacks.get(args.id).to_dict()
 | |
|                 resolved = True
 | |
|             except exc.HTTPNotFound:
 | |
|                 raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|         for output in outputs['outputs']:
 | |
|             if resolved:
 | |
|                 show_output({'output': output})
 | |
|             else:
 | |
|                 show_output(resolve_output(output['output_key']))
 | |
|     else:
 | |
|         show_output(resolve_output(args.output))
 | |
| 
 | |
| 
 | |
| @utils.arg('-f', '--filters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Filter parameters to apply on returned resource types. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by a semicolon. It can be any of '
 | |
|                   'name, version and support_status'),
 | |
|            action='append')
 | |
| def do_resource_type_list(hc, args):
 | |
|     '''List the available resource types.'''
 | |
|     types = hc.resource_types.list(
 | |
|         filters=utils.format_parameters(args.filters))
 | |
|     utils.print_list(types, ['resource_type'], sortby_index=0)
 | |
| 
 | |
| 
 | |
| @utils.arg('resource_type', metavar='<RESOURCE_TYPE>',
 | |
|            help=_('Resource type to get the details for.'))
 | |
| def do_resource_type_show(hc, args):
 | |
|     '''Show the resource type.'''
 | |
|     try:
 | |
|         resource_type = hc.resource_types.get(args.resource_type)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(
 | |
|             _('Resource Type not found: %s') % args.resource_type)
 | |
|     else:
 | |
|         print(jsonutils.dumps(resource_type, indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('resource_type', metavar='<RESOURCE_TYPE>',
 | |
|            help=_('Resource type to generate a template for.'))
 | |
| @utils.arg('-t', '--template-type', metavar='<TEMPLATE_TYPE>',
 | |
|            default='cfn',
 | |
|            help=_('Template type to generate, hot or cfn.'))
 | |
| @utils.arg('-F', '--format', metavar='<FORMAT>',
 | |
|            help=_("The template output format, one of: %s.")
 | |
|                  % ', '.join(utils.supported_formats.keys()))
 | |
| def do_resource_type_template(hc, args):
 | |
|     '''Generate a template based on a resource type.'''
 | |
|     fields = {'resource_type': args.resource_type,
 | |
|               'template_type': args.template_type}
 | |
|     try:
 | |
|         template = hc.resource_types.generate_template(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(
 | |
|             _('Resource Type %s not found.') % args.resource_type)
 | |
|     else:
 | |
|         if args.format:
 | |
|             print(utils.format_output(template, format=args.format))
 | |
|         else:
 | |
|             print(utils.format_output(template))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to get the template for.'))
 | |
| def do_template_show(hc, args):
 | |
|     '''Get the template for the specified stack.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         template = hc.stacks.template(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         if 'heat_template_version' in template:
 | |
|             print(yaml.safe_dump(template, indent=2))
 | |
|         else:
 | |
|             print(jsonutils.dumps(template, indent=2, ensure_ascii=False))
 | |
| 
 | |
| 
 | |
| @utils.arg('-u', '--template-url', metavar='<URL>',
 | |
|            help=_('URL of template.'))
 | |
| @utils.arg('-f', '--template-file', metavar='<FILE>',
 | |
|            help=_('Path to the template.'))
 | |
| @utils.arg('-e', '--environment-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to the environment, it can be specified '
 | |
|                   'multiple times.'),
 | |
|            action='append')
 | |
| @utils.arg('-o', '--template-object', metavar='<URL>',
 | |
|            help=_('URL to retrieve template object (e.g. from swift).'))
 | |
| @utils.arg('-n', '--show-nested', default=False, action="store_true",
 | |
|            help=_('Resolve parameters from nested templates as well.'))
 | |
| @utils.arg('-P', '--parameters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Parameter values for the template. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by a semicolon.'),
 | |
|            action='append')
 | |
| def do_template_validate(hc, args):
 | |
|     """Validate a template with parameters."""
 | |
| 
 | |
|     tpl_files, template = template_utils.get_template_contents(
 | |
|         args.template_file,
 | |
|         args.template_url,
 | |
|         args.template_object,
 | |
|         _authenticated_fetcher(hc))
 | |
| 
 | |
|     env_files, env = template_utils.process_multiple_environments_and_files(
 | |
|         env_paths=args.environment_file)
 | |
|     fields = {
 | |
|         'template': template,
 | |
|         'parameters': utils.format_parameters(args.parameters),
 | |
|         'files': dict(list(tpl_files.items()) + list(env_files.items())),
 | |
|         'environment': env,
 | |
|     }
 | |
| 
 | |
|     if args.show_nested:
 | |
|         fields['show_nested'] = args.show_nested
 | |
| 
 | |
|     validation = hc.stacks.validate(**fields)
 | |
|     print(jsonutils.dumps(validation, indent=2, ensure_ascii=False))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the resources for.'))
 | |
| @utils.arg('-n', '--nested-depth', metavar='<DEPTH>',
 | |
|            help=_('Depth of nested stacks from which to display resources.'))
 | |
| @utils.arg('--with-detail', default=False, action="store_true",
 | |
|            help=_('Enable detail information presented for each resource '
 | |
|                   'in resources list.'))
 | |
| def do_resource_list(hc, args):
 | |
|     '''Show list of resources belonging to a stack.'''
 | |
|     fields = {
 | |
|         'stack_id': args.id,
 | |
|         'nested_depth': args.nested_depth,
 | |
|         'with_detail': args.with_detail,
 | |
|     }
 | |
|     try:
 | |
|         resources = hc.resources.list(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         fields = ['physical_resource_id', 'resource_type',
 | |
|                   'resource_status', 'updated_time']
 | |
|         if len(resources) >= 1 and not hasattr(resources[0], 'resource_name'):
 | |
|             fields.insert(0, 'logical_resource_id')
 | |
|         else:
 | |
|             fields.insert(0, 'resource_name')
 | |
| 
 | |
|         if args.nested_depth or args.with_detail:
 | |
|             fields.append('stack_name')
 | |
| 
 | |
|         utils.print_list(resources, fields, sortby_index=4)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the resource for.'))
 | |
| @utils.arg('resource', metavar='<RESOURCE>',
 | |
|            help=_('Name of the resource to show the details for.'))
 | |
| @utils.arg('-a', '--with-attr', metavar='<ATTRIBUTE>',
 | |
|            help=_('Attribute to show, it can be specified '
 | |
|                   'multiple times.'),
 | |
|            action='append')
 | |
| def do_resource_show(hc, args):
 | |
|     '''Describe the resource.'''
 | |
|     fields = {'stack_id': args.id,
 | |
|               'resource_name': args.resource}
 | |
|     if args.with_attr:
 | |
|         fields['with_attr'] = list(args.with_attr)
 | |
|     try:
 | |
|         resource = hc.resources.get(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack or resource not found: '
 | |
|                                  '%(id)s %(resource)s') %
 | |
|                                {'id': args.id, 'resource': args.resource})
 | |
|     else:
 | |
|         formatters = {
 | |
|             'attributes': utils.json_formatter,
 | |
|             'links': utils.link_formatter,
 | |
|             'required_by': utils.newline_list_formatter
 | |
|         }
 | |
|         utils.print_dict(resource.to_dict(), formatters=formatters)
 | |
| 
 | |
| 
 | |
| @utils.arg('resource_type', metavar='<RESOURCE_TYPE>',
 | |
|            help=_('Resource type to generate a template for.'))
 | |
| @utils.arg('-t', '--template-type', metavar='<TEMPLATE_TYPE>',
 | |
|            default='cfn',
 | |
|            help=_('Template type to generate, hot or cfn.'))
 | |
| @utils.arg('-F', '--format', metavar='<FORMAT>',
 | |
|            help=_("The template output format, one of: %s.")
 | |
|                  % ', '.join(utils.supported_formats.keys()))
 | |
| def do_resource_template(hc, args):
 | |
|     '''DEPRECATED! Use resource-type-template instead.'''
 | |
|     logger.warning(_LW('DEPRECATED! Use %(cmd)s instead.'),
 | |
|                    {'cmd': 'resource-type-template'})
 | |
|     do_resource_type_template(hc, args)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the resource metadata for.'))
 | |
| @utils.arg('resource', metavar='<RESOURCE>',
 | |
|            help=_('Name of the resource to show the metadata for.'))
 | |
| def do_resource_metadata(hc, args):
 | |
|     '''List resource metadata.'''
 | |
|     fields = {'stack_id': args.id,
 | |
|               'resource_name': args.resource}
 | |
|     try:
 | |
|         metadata = hc.resources.metadata(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack or resource not found: '
 | |
|                                  '%(id)s %(resource)s') %
 | |
|                                {'id': args.id, 'resource': args.resource})
 | |
|     else:
 | |
|         print(jsonutils.dumps(metadata, indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack the resource belongs to.'))
 | |
| @utils.arg('resource', metavar='<RESOURCE>',
 | |
|            help=_('Name of the resource to signal.'))
 | |
| @utils.arg('-D', '--data', metavar='<DATA>',
 | |
|            help=_('JSON Data to send to the signal handler.'))
 | |
| @utils.arg('-f', '--data-file', metavar='<FILE>',
 | |
|            help=_('File containing JSON data to send to the signal handler.'))
 | |
| def do_resource_signal(hc, args):
 | |
|     '''Send a signal to a resource.'''
 | |
|     fields = {'stack_id': args.id,
 | |
|               'resource_name': args.resource}
 | |
|     data = args.data
 | |
|     data_file = args.data_file
 | |
|     if data and data_file:
 | |
|         raise exc.CommandError(_('Can only specify one of data and data-file'))
 | |
|     if data_file:
 | |
|         data_url = utils.normalise_file_path_to_url(data_file)
 | |
|         data = request.urlopen(data_url).read()
 | |
|     if data:
 | |
|         if isinstance(data, six.binary_type):
 | |
|             data = data.decode('utf-8')
 | |
|         try:
 | |
|             data = jsonutils.loads(data)
 | |
|         except ValueError as ex:
 | |
|             raise exc.CommandError(_('Data should be in JSON format: %s') % ex)
 | |
|         if not isinstance(data, dict):
 | |
|             raise exc.CommandError(_('Data should be a JSON dict'))
 | |
|         fields['data'] = data
 | |
|     try:
 | |
|         hc.resources.signal(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack or resource not found: '
 | |
|                                  '%(id)s %(resource)s') %
 | |
|                                {'id': args.id, 'resource': args.resource})
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of the stack these resources belong to.'))
 | |
| @utils.arg('--pre-create', action='store_true', default=False,
 | |
|            help=_('Clear the pre-create hooks (optional)'))
 | |
| @utils.arg('--pre-update', action='store_true', default=False,
 | |
|            help=_('Clear the pre-update hooks (optional)'))
 | |
| @utils.arg('hook', metavar='<RESOURCE>', nargs='+',
 | |
|            help=_('Resource names with hooks to clear. Resources '
 | |
|                   'in nested stacks can be set using slash as a separator: '
 | |
|                   'nested_stack/another/my_resource. You can use wildcards '
 | |
|                   'to match multiple stacks or resources: '
 | |
|                   'nested_stack/an*/*_resource'))
 | |
| def do_hook_clear(hc, args):
 | |
|     '''Clear hooks on a given stack.'''
 | |
|     if args.pre_create:
 | |
|         hook_type = 'pre-create'
 | |
|     elif args.pre_update:
 | |
|         hook_type = 'pre-update'
 | |
|     else:
 | |
|         hook_type = _get_hook_type_via_status(hc, args.id)
 | |
| 
 | |
|     for hook_string in args.hook:
 | |
|         hook = [b for b in hook_string.split('/') if b]
 | |
|         resource_pattern = hook[-1]
 | |
|         stack_id = args.id
 | |
| 
 | |
|         def clear_hook(stack_id, resource_name):
 | |
|             try:
 | |
|                 hc.resources.signal(
 | |
|                     stack_id=stack_id,
 | |
|                     resource_name=resource_name,
 | |
|                     data={'unset_hook': hook_type})
 | |
|             except exc.HTTPNotFound:
 | |
|                 logger.error(
 | |
|                     _LE("Stack %(stack)s or resource %(resource)s not found"),
 | |
|                     {'resource': resource_name, 'stack': stack_id})
 | |
| 
 | |
|         def clear_wildcard_hooks(stack_id, stack_patterns):
 | |
|             if stack_patterns:
 | |
|                 for resource in hc.resources.list(stack_id):
 | |
|                     res_name = resource.resource_name
 | |
|                     if fnmatch.fnmatchcase(res_name, stack_patterns[0]):
 | |
|                         nested_stack = hc.resources.get(
 | |
|                             stack_id=stack_id,
 | |
|                             resource_name=res_name)
 | |
|                         clear_wildcard_hooks(
 | |
|                             nested_stack.physical_resource_id,
 | |
|                             stack_patterns[1:])
 | |
|             else:
 | |
|                 for resource in hc.resources.list(stack_id):
 | |
|                     res_name = resource.resource_name
 | |
|                     if fnmatch.fnmatchcase(res_name, resource_pattern):
 | |
|                         clear_hook(stack_id, res_name)
 | |
| 
 | |
|         clear_wildcard_hooks(stack_id, hook[:-1])
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the events for.'))
 | |
| @utils.arg('-r', '--resource', metavar='<RESOURCE>',
 | |
|            help=_('Name of the resource to filter events by.'))
 | |
| @utils.arg('-f', '--filters', metavar='<KEY1=VALUE1;KEY2=VALUE2...>',
 | |
|            help=_('Filter parameters to apply on returned events. '
 | |
|                   'This can be specified multiple times, or once with '
 | |
|                   'parameters separated by a semicolon.'),
 | |
|            action='append')
 | |
| @utils.arg('-l', '--limit', metavar='<LIMIT>',
 | |
|            help=_('Limit the number of events returned.'))
 | |
| @utils.arg('-m', '--marker', metavar='<ID>',
 | |
|            help=_('Only return events that appear after the given event ID.'))
 | |
| @utils.arg('-n', '--nested-depth', metavar='<DEPTH>',
 | |
|            help=_('Depth of nested stacks from which to display events. '
 | |
|                   'Note this cannot be specified with --resource.'))
 | |
| @utils.arg('-F', '--format', metavar='<FORMAT>',
 | |
|            help=_('The output value format, one of: log, table'),
 | |
|            default='table')
 | |
| def do_event_list(hc, args):
 | |
|     '''List events for a stack.'''
 | |
|     display_fields = ['id', 'resource_status_reason',
 | |
|                       'resource_status', 'event_time']
 | |
|     event_args = {'resource_name': args.resource,
 | |
|                   'limit': args.limit,
 | |
|                   'marker': args.marker,
 | |
|                   'filters': utils.format_parameters(args.filters),
 | |
|                   'sort_dir': 'asc'}
 | |
| 
 | |
|     # Specifying a resource in recursive mode makes no sense..
 | |
|     if args.nested_depth and args.resource:
 | |
|         msg = _("--nested-depth cannot be specified with --resource")
 | |
|         raise exc.CommandError(msg)
 | |
| 
 | |
|     if args.nested_depth:
 | |
|         try:
 | |
|             nested_depth = int(args.nested_depth)
 | |
|         except ValueError:
 | |
|             msg = _("--nested-depth invalid value %s") % args.nested_depth
 | |
|             raise exc.CommandError(msg)
 | |
|         # Until the API supports recursive event listing we'll have to do the
 | |
|         # marker/limit filtering client-side
 | |
|         del (event_args['marker'])
 | |
|         del (event_args['limit'])
 | |
|         # Nested list adds the stack name to the output
 | |
|         display_fields.append('stack_name')
 | |
|     else:
 | |
|         nested_depth = 0
 | |
| 
 | |
|     events = event_utils.get_events(
 | |
|         hc, stack_id=args.id, event_args=event_args, nested_depth=nested_depth,
 | |
|         marker=args.marker, limit=args.limit)
 | |
| 
 | |
|     if len(events) >= 1:
 | |
|         if hasattr(events[0], 'resource_name'):
 | |
|             display_fields.insert(0, 'resource_name')
 | |
|         else:
 | |
|             display_fields.insert(0, 'logical_resource_id')
 | |
| 
 | |
|     if args.format == 'log':
 | |
|         print(utils.event_log_formatter(events))
 | |
|     else:
 | |
|         utils.print_list(events, display_fields, sortby_index=None)
 | |
| 
 | |
| 
 | |
| def _get_hook_type_via_status(hc, stack_id):
 | |
|     # Figure out if the hook should be pre-create or pre-update based
 | |
|     # on the stack status, also sanity assertions that we're in-progress.
 | |
|     try:
 | |
|         stack = hc.stacks.get(stack_id=stack_id)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % stack_id)
 | |
|     else:
 | |
|         if 'IN_PROGRESS' not in stack.stack_status:
 | |
|             raise exc.CommandError(_('Stack status %s not IN_PROGRESS') %
 | |
|                                    stack.stack_status)
 | |
| 
 | |
|     if 'CREATE' in stack.stack_status:
 | |
|         hook_type = 'pre-create'
 | |
|     elif 'UPDATE' in stack.stack_status:
 | |
|         hook_type = 'pre-update'
 | |
|     else:
 | |
|         raise exc.CommandError(_('Unexpected stack status %s, '
 | |
|                                  'only create/update supported')
 | |
|                                % stack.stack_status)
 | |
|     return hook_type
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the pending hooks for.'))
 | |
| @utils.arg('-n', '--nested-depth', metavar='<DEPTH>',
 | |
|            help=_('Depth of nested stacks from which to display hooks.'))
 | |
| def do_hook_poll(hc, args):
 | |
|     '''List resources with pending hook for a stack.'''
 | |
| 
 | |
|     # There are a few steps to determining if a stack has pending hooks
 | |
|     # 1. The stack is IN_PROGRESS status (otherwise, by definition no hooks
 | |
|     #    can be pending
 | |
|     # 2. There is an event for a resource associated with hitting a hook
 | |
|     # 3. There is not an event associated with clearing the hook in step(2)
 | |
|     #
 | |
|     # So, essentially, this ends up being a specially filtered type of event
 | |
|     # listing, because all hook status is exposed via events.  In future
 | |
|     # we might consider exposing some more efficient interface via the API
 | |
|     # to reduce the expense of this brute-force polling approach
 | |
|     display_fields = ['id', 'resource_status_reason',
 | |
|                       'resource_status', 'event_time']
 | |
|     if args.nested_depth:
 | |
|         try:
 | |
|             nested_depth = int(args.nested_depth)
 | |
|         except ValueError:
 | |
|             msg = _("--nested-depth invalid value %s") % args.nested_depth
 | |
|             raise exc.CommandError(msg)
 | |
|         display_fields.append('stack_name')
 | |
|     else:
 | |
|         nested_depth = 0
 | |
| 
 | |
|     hook_type = _get_hook_type_via_status(hc, args.id)
 | |
|     event_args = {'sort_dir': 'asc'}
 | |
|     hook_events = event_utils.get_hook_events(
 | |
|         hc, stack_id=args.id, event_args=event_args,
 | |
|         nested_depth=nested_depth, hook_type=hook_type)
 | |
| 
 | |
|     if len(hook_events) >= 1:
 | |
|         if hasattr(hook_events[0], 'resource_name'):
 | |
|             display_fields.insert(0, 'resource_name')
 | |
|         else:
 | |
|             display_fields.insert(0, 'logical_resource_id')
 | |
| 
 | |
|     utils.print_list(hook_events, display_fields, sortby_index=None)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the events for.'))
 | |
| @utils.arg('resource', metavar='<RESOURCE>',
 | |
|            help=_('Name of the resource the event belongs to.'))
 | |
| @utils.arg('event', metavar='<EVENT>',
 | |
|            help=_('ID of event to display details for.'))
 | |
| def do_event(hc, args):
 | |
|     '''DEPRECATED! Use event-show instead.'''
 | |
|     logger.warning(_LW('DEPRECATED! Use %(cmd)s instead.'),
 | |
|                    {'cmd': 'event-show'})
 | |
|     do_event_show(hc, args)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to show the events for.'))
 | |
| @utils.arg('resource', metavar='<RESOURCE>',
 | |
|            help=_('Name of the resource the event belongs to.'))
 | |
| @utils.arg('event', metavar='<EVENT>',
 | |
|            help=_('ID of event to display details for.'))
 | |
| def do_event_show(hc, args):
 | |
|     '''Describe the event.'''
 | |
|     fields = {'stack_id': args.id,
 | |
|               'resource_name': args.resource,
 | |
|               'event_id': args.event}
 | |
|     try:
 | |
|         event = hc.events.get(**fields)
 | |
|     except exc.HTTPNotFound as ex:
 | |
|         # it could be the stack/resource/or event that is not found
 | |
|         # just use the message that the server sent us.
 | |
|         raise exc.CommandError(str(ex))
 | |
|     else:
 | |
|         formatters = {
 | |
|             'links': utils.link_formatter,
 | |
|             'resource_properties': utils.json_formatter
 | |
|         }
 | |
|         utils.print_dict(event.to_dict(), formatters=formatters)
 | |
| 
 | |
| 
 | |
| @utils.arg('-f', '--definition-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to JSON/YAML containing map defining '
 | |
|                   '<inputs>, <outputs>, and <options>.'))
 | |
| @utils.arg('-c', '--config-file', metavar='<FILE or URL>',
 | |
|            help=_('Path to configuration script/data.'))
 | |
| @utils.arg('-g', '--group', metavar='<GROUP_NAME>', default='Heat::Ungrouped',
 | |
|            help=_('Group name of configuration tool expected by the config.'))
 | |
| @utils.arg('name', metavar='<CONFIG_NAME>',
 | |
|            help=_('Name of the configuration to create.'))
 | |
| def do_config_create(hc, args):
 | |
|     '''Create a software configuration.'''
 | |
|     config = {
 | |
|         'group': args.group,
 | |
|         'config': ''
 | |
|     }
 | |
| 
 | |
|     defn = {}
 | |
|     if args.definition_file:
 | |
|         defn_url = utils.normalise_file_path_to_url(
 | |
|             args.definition_file)
 | |
|         defn_raw = request.urlopen(defn_url).read() or '{}'
 | |
|         defn = yaml.load(defn_raw, Loader=template_format.yaml_loader)
 | |
| 
 | |
|     config['inputs'] = defn.get('inputs', [])
 | |
|     config['outputs'] = defn.get('outputs', [])
 | |
|     config['options'] = defn.get('options', {})
 | |
| 
 | |
|     if args.config_file:
 | |
|         config_url = utils.normalise_file_path_to_url(
 | |
|             args.config_file)
 | |
|         config['config'] = request.urlopen(config_url).read()
 | |
| 
 | |
|     # build a mini-template with a config resource and validate it
 | |
|     validate_template = {
 | |
|         'heat_template_version': '2013-05-23',
 | |
|         'resources': {
 | |
|             args.name: {
 | |
|                 'type': 'OS::Heat::SoftwareConfig',
 | |
|                 'properties': config
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     hc.stacks.validate(template=validate_template)
 | |
| 
 | |
|     config['name'] = args.name
 | |
| 
 | |
|     sc = hc.software_configs.create(**config)
 | |
|     print(jsonutils.dumps(sc.to_dict(), indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('-l', '--limit', metavar='<LIMIT>',
 | |
|            help=_('Limit the number of configs returned.'))
 | |
| @utils.arg('-m', '--marker', metavar='<ID>',
 | |
|            help=_('Return configs that appear after the given config ID.'))
 | |
| def do_config_list(hc, args):
 | |
|     '''List software configs.'''
 | |
|     kwargs = {}
 | |
|     if args.limit:
 | |
|         kwargs['limit'] = args.limit
 | |
|     if args.marker:
 | |
|         kwargs['marker'] = args.marker
 | |
|     scs = hc.software_configs.list(**kwargs)
 | |
|     fields = ['id', 'name', 'group', 'creation_time']
 | |
|     utils.print_list(scs, fields, sortby_index=None)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<ID>',
 | |
|            help=_('ID of the config.'))
 | |
| @utils.arg('-c', '--config-only', default=False, action="store_true",
 | |
|            help=_('Only display the value of the <config> property.'))
 | |
| def do_config_show(hc, args):
 | |
|     '''View details of a software configuration.'''
 | |
|     try:
 | |
|         sc = hc.software_configs.get(config_id=args.id)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError('Configuration not found: %s' % args.id)
 | |
|     else:
 | |
|         if args.config_only:
 | |
|             print(sc.config)
 | |
|         else:
 | |
|             print(jsonutils.dumps(sc.to_dict(), indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<ID>', nargs='+',
 | |
|            help=_('IDs of the configurations to delete.'))
 | |
| def do_config_delete(hc, args):
 | |
|     '''Delete a software configuration.'''
 | |
|     failure_count = 0
 | |
| 
 | |
|     for config_id in args.id:
 | |
|         try:
 | |
|             hc.software_configs.delete(config_id=config_id)
 | |
|         except exc.HTTPNotFound as e:
 | |
|             failure_count += 1
 | |
|             print(e)
 | |
|     if failure_count == len(args.id):
 | |
|         raise exc.CommandError(_("Unable to delete any of the specified "
 | |
|                                  "configs."))
 | |
| 
 | |
| 
 | |
| @utils.arg('-i', '--input-value', metavar='<KEY=VALUE>',
 | |
|            help=_('Input value to set on the deployment. '
 | |
|                   'This can be specified multiple times.'),
 | |
|            action='append')
 | |
| @utils.arg('-a', '--action', metavar='<ACTION>', default='UPDATE',
 | |
|            help=_('Name of action for this deployment. '
 | |
|                   'Can be a custom action, or one of: '
 | |
|                   'CREATE, UPDATE, DELETE, SUSPEND, RESUME'))
 | |
| @utils.arg('-c', '--config', metavar='<CONFIG>',
 | |
|            help=_('ID of the configuration to deploy.'))
 | |
| @utils.arg('-s', '--server', metavar='<SERVER>', required=True,
 | |
|            help=_('ID of the server being deployed to.'))
 | |
| @utils.arg('-t', '--signal-transport',
 | |
|            default='TEMP_URL_SIGNAL',
 | |
|            metavar='<TRANSPORT>',
 | |
|            help=_('How the server should signal to heat with the deployment '
 | |
|                   'output values. TEMP_URL_SIGNAL will create a '
 | |
|                   'Swift TempURL to be signaled via HTTP PUT. NO_SIGNAL will '
 | |
|                   'result in the resource going to the COMPLETE state '
 | |
|                   'without waiting for any signal.'))
 | |
| @utils.arg('--container', metavar='<CONTAINER_NAME>',
 | |
|            help=_('Optional name of container to store TEMP_URL_SIGNAL '
 | |
|                   'objects in. If not specified a container will be created '
 | |
|                   'with a name derived from the DEPLOY_NAME'))
 | |
| @utils.arg('--timeout', metavar='<TIMEOUT>',
 | |
|            type=int,
 | |
|            default=60,
 | |
|            help=_('Deployment timeout in minutes.'))
 | |
| @utils.arg('name', metavar='<DEPLOY_NAME>',
 | |
|            help=_('Name of the derived config associated with this '
 | |
|                   'deployment. This is used to apply a sort order to the '
 | |
|                   'list of configurations currently deployed to the server.'))
 | |
| def do_deployment_create(hc, args):
 | |
|     '''Create a software deployment.'''
 | |
|     config = {}
 | |
|     if args.config:
 | |
|         try:
 | |
|             config = hc.software_configs.get(config_id=args.config)
 | |
|         except exc.HTTPNotFound:
 | |
|             raise exc.CommandError(
 | |
|                 _('Configuration not found: %s') % args.config)
 | |
| 
 | |
|     derrived_params = deployment_utils.build_derived_config_params(
 | |
|         action=args.action,
 | |
|         source=config,
 | |
|         name=args.name,
 | |
|         input_values=utils.format_parameters(args.input_value, False),
 | |
|         server_id=args.server,
 | |
|         signal_transport=args.signal_transport,
 | |
|         signal_id=deployment_utils.build_signal_id(hc, args)
 | |
|     )
 | |
|     derived_config = hc.software_configs.create(**derrived_params)
 | |
| 
 | |
|     sd = hc.software_deployments.create(
 | |
|         tenant_id='asdf',
 | |
|         config_id=derived_config.id,
 | |
|         server_id=args.server,
 | |
|         action=args.action,
 | |
|         status='IN_PROGRESS'
 | |
|     )
 | |
|     print(jsonutils.dumps(sd.to_dict(), indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('-s', '--server', metavar='<SERVER>',
 | |
|            help=_('ID of the server to fetch deployments for.'))
 | |
| def do_deployment_list(hc, args):
 | |
|     '''List software deployments.'''
 | |
|     kwargs = {'server_id': args.server} if args.server else {}
 | |
|     deployments = hc.software_deployments.list(**kwargs)
 | |
|     fields = ['id', 'config_id', 'server_id', 'action', 'status',
 | |
|               'creation_time', 'status_reason']
 | |
|     utils.print_list(deployments, fields, sortby_index=5)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<ID>',
 | |
|            help=_('ID of the deployment.'))
 | |
| def do_deployment_show(hc, args):
 | |
|     '''Show the details of a software deployment.'''
 | |
|     try:
 | |
|         sd = hc.software_deployments.get(deployment_id=args.id)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Deployment not found: %s') % args.id)
 | |
|     else:
 | |
|         print(jsonutils.dumps(sd.to_dict(), indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<ID>',
 | |
|            help=_('ID of the server to fetch deployments for.'))
 | |
| def do_deployment_metadata_show(hc, args):
 | |
|     '''Get deployment configuration metadata for the specified server.'''
 | |
|     md = hc.software_deployments.metadata(server_id=args.id)
 | |
|     print(jsonutils.dumps(md, indent=2))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<ID>', nargs='+',
 | |
|            help=_('IDs of the deployments to delete.'))
 | |
| def do_deployment_delete(hc, args):
 | |
|     '''Delete software deployments.'''
 | |
|     failure_count = 0
 | |
| 
 | |
|     for deploy_id in args.id:
 | |
|         try:
 | |
|             sd = hc.software_deployments.get(deployment_id=deploy_id)
 | |
|             hc.software_deployments.delete(deployment_id=deploy_id)
 | |
|         except Exception as e:
 | |
|             if isinstance(e, exc.HTTPNotFound):
 | |
|                 print(_('Deployment with ID %s not found') % deploy_id)
 | |
|             failure_count += 1
 | |
|             continue
 | |
| 
 | |
|         # just try best to delete the corresponding config
 | |
|         try:
 | |
|             config_id = getattr(sd, 'config_id')
 | |
|             hc.software_configs.delete(config_id=config_id)
 | |
|         except Exception:
 | |
|             print(_('Failed to delete the correlative config '
 | |
|                     '%(config_id)s of deployment %(deploy_id)s') %
 | |
|                   {'config_id': config_id, 'deploy_id': deploy_id})
 | |
| 
 | |
|     if failure_count == len(args.id):
 | |
|         raise exc.CommandError(_("Unable to delete any of the specified "
 | |
|                                  "deployments."))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<ID>',
 | |
|            help=_('ID deployment to show the output for.'))
 | |
| @utils.arg('output', metavar='<OUTPUT NAME>', nargs='?', default=None,
 | |
|            help=_('Name of an output to display.'))
 | |
| @utils.arg('-a', '--all', default=False, action='store_true',
 | |
|            help=_('Display all deployment outputs.'))
 | |
| @utils.arg('-F', '--format', metavar='<FORMAT>',
 | |
|            help=_('The output value format, one of: raw, json'),
 | |
|            default='raw')
 | |
| def do_deployment_output_show(hc, args):
 | |
|     '''Show a specific deployment output.'''
 | |
|     if (not args.all and args.output is None or
 | |
|             args.all and args.output is not None):
 | |
|         raise exc.CommandError(
 | |
|             _('Error: either %(output)s or %(all)s argument is needed.')
 | |
|             % {'output': '<OUTPUT NAME>', 'all': '--all'})
 | |
|     try:
 | |
|         sd = hc.software_deployments.get(deployment_id=args.id)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Deployment not found: %s') % args.id)
 | |
|     outputs = sd.to_dict().get('output_values', {})
 | |
| 
 | |
|     if args.all:
 | |
|         print(utils.json_formatter(outputs))
 | |
|     else:
 | |
|         for output_key, value in outputs.items():
 | |
|             if output_key == args.output:
 | |
|                 break
 | |
|         else:
 | |
|             return
 | |
| 
 | |
|         if (args.format == 'json'
 | |
|                 or isinstance(value, dict)
 | |
|                 or isinstance(value, list)):
 | |
|             print(utils.json_formatter(value))
 | |
|         else:
 | |
|             print(value)
 | |
| 
 | |
| 
 | |
| def do_build_info(hc, args):
 | |
|     '''Retrieve build information.'''
 | |
|     result = hc.build_info.build_info()
 | |
|     formatters = {
 | |
|         'api': utils.json_formatter,
 | |
|         'engine': utils.json_formatter,
 | |
|     }
 | |
|     utils.print_dict(result, formatters=formatters)
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of stack to snapshot.'))
 | |
| @utils.arg('-n', '--name', metavar='<NAME>',
 | |
|            help=_('If specified, the name given to the snapshot.'))
 | |
| def do_stack_snapshot(hc, args):
 | |
|     '''Make a snapshot of a stack.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     if args.name:
 | |
|         fields['name'] = args.name
 | |
|     try:
 | |
|         snapshot = hc.stacks.snapshot(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         print(jsonutils.dumps(snapshot, indent=2, ensure_ascii=False))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of the stack containing the snapshot.'))
 | |
| @utils.arg('snapshot', metavar='<SNAPSHOT>',
 | |
|            help=_('The ID of the snapshot to show.'))
 | |
| def do_snapshot_show(hc, args):
 | |
|     '''Show a snapshot of a stack.'''
 | |
|     fields = {'stack_id': args.id, 'snapshot_id': args.snapshot}
 | |
|     try:
 | |
|         snapshot = hc.stacks.snapshot_show(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack or snapshot not found'))
 | |
|     else:
 | |
|         print(jsonutils.dumps(snapshot, indent=2, ensure_ascii=False))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of the stack containing the snapshot.'))
 | |
| @utils.arg('snapshot', metavar='<SNAPSHOT>',
 | |
|            help=_('The ID of the snapshot to delete.'))
 | |
| def do_snapshot_delete(hc, args):
 | |
|     '''Delete a snapshot of a stack.'''
 | |
|     fields = {'stack_id': args.id, 'snapshot_id': args.snapshot}
 | |
|     try:
 | |
|         hc.stacks.snapshot_delete(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack or snapshot not found'))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of the stack containing the snapshot.'))
 | |
| @utils.arg('snapshot', metavar='<SNAPSHOT>',
 | |
|            help=_('The ID of the snapshot to restore.'))
 | |
| def do_stack_restore(hc, args):
 | |
|     '''Restore a snapshot of a stack.'''
 | |
|     fields = {'stack_id': args.id, 'snapshot_id': args.snapshot}
 | |
|     try:
 | |
|         hc.stacks.restore(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack or snapshot not found'))
 | |
| 
 | |
| 
 | |
| @utils.arg('id', metavar='<NAME or ID>',
 | |
|            help=_('Name or ID of the stack containing the snapshots.'))
 | |
| def do_snapshot_list(hc, args):
 | |
|     '''List the snapshots of a stack.'''
 | |
|     fields = {'stack_id': args.id}
 | |
|     try:
 | |
|         snapshots = hc.stacks.snapshot_list(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') % args.id)
 | |
|     else:
 | |
|         fields = ['id', 'name', 'status', 'status_reason', 'creation_time']
 | |
|         formatters = {
 | |
|             'id': lambda x: x['id'],
 | |
|             'name': lambda x: x['name'],
 | |
|             'status': lambda x: x['status'],
 | |
|             'status_reason': lambda x: x['status_reason'],
 | |
|             'creation_time': lambda x: x['creation_time'],
 | |
|         }
 | |
|         utils.print_list(snapshots["snapshots"], fields, formatters=formatters)
 | |
| 
 | |
| 
 | |
| def do_service_list(hc, args=None):
 | |
|     '''List the Heat engines.'''
 | |
|     fields = ['hostname', 'binary', 'engine_id', 'host',
 | |
|               'topic', 'updated_at', 'status']
 | |
|     services = hc.services.list()
 | |
|     utils.print_list(services, fields, sortby_index=1)
 | |
| 
 | |
| 
 | |
| def do_template_version_list(hc, args):
 | |
|     '''List the available template versions.'''
 | |
|     versions = hc.template_versions.list()
 | |
|     fields = ['version', 'type']
 | |
|     utils.print_list(versions, fields, sortby_index=1)
 | |
| 
 | |
| 
 | |
| @utils.arg('template_version', metavar='<TEMPLATE_VERSION>',
 | |
|            help=_('Template version to get the functions for.'))
 | |
| def do_template_function_list(hc, args):
 | |
|     '''List the available functions.'''
 | |
|     try:
 | |
|         functions = hc.template_versions.get(args.template_version)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(
 | |
|             _('Template version not found: %s') % args.template_version)
 | |
|     else:
 | |
|         utils.print_list(functions, ['functions', 'description'])
 | |
| 
 | |
| 
 | |
| def _do_stack_show(hc, fields):
 | |
|     try:
 | |
|         stack = hc.stacks.get(**fields)
 | |
|     except exc.HTTPNotFound:
 | |
|         raise exc.CommandError(_('Stack not found: %s') %
 | |
|                                fields.get('stack_id'))
 | |
|     else:
 | |
|         formatters = {
 | |
|             'description': utils.text_wrap_formatter,
 | |
|             'template_description': utils.text_wrap_formatter,
 | |
|             'stack_status_reason': utils.text_wrap_formatter,
 | |
|             'parameters': utils.json_formatter,
 | |
|             'outputs': utils.json_formatter,
 | |
|             'links': utils.link_formatter,
 | |
|             'tags': utils.json_formatter
 | |
|         }
 | |
|         utils.print_dict(stack.to_dict(), formatters=formatters)
 | |
| 
 | |
| 
 | |
| def _poll_for_events(hc, stack_name, action, poll_period):
 | |
|     """Continuously poll events and logs for performed action on stack."""
 | |
| 
 | |
|     fields = {'stack_id': stack_name}
 | |
|     _do_stack_show(hc, fields)
 | |
|     marker = None
 | |
|     while True:
 | |
|         events = event_utils.get_events(hc, stack_id=stack_name,
 | |
|                                         event_args={'sort_dir': 'asc',
 | |
|                                                     'marker': marker})
 | |
| 
 | |
|         if len(events) >= 1:
 | |
|             # set marker to last event that was received.
 | |
|             marker = getattr(events[-1], 'id', None)
 | |
|             events_log = utils.event_log_formatter(events)
 | |
|             print(events_log)
 | |
|             for event in events:
 | |
|                 # check if stack event was also received
 | |
|                 if getattr(event, 'resource_name', '') == stack_name:
 | |
|                     stack_status = getattr(event, 'resource_status', '')
 | |
|                     msg = _("\n Stack %(name)s %(status)s \n") % dict(
 | |
|                         name=stack_name, status=stack_status)
 | |
|                     if stack_status == '%s_COMPLETE' % action:
 | |
|                         _do_stack_show(hc, fields)
 | |
|                         print(msg)
 | |
|                         return
 | |
|                     elif stack_status == '%s_FAILED' % action:
 | |
|                         _do_stack_show(hc, fields)
 | |
|                         raise exc.StackFailure(msg)
 | |
| 
 | |
|         time.sleep(poll_period)
 |