mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 13:34:01 -07:00 
			
		
		
		
	* Start of migration to argparse * various fixes and improvements * Linting fixes * Test fixes * Fix vault_password_files * Add PrependAction for argparse * A bunch of additional tweak/fixes * Fix ansible-config tests * Fix man page generation * linting fix * More adhoc pattern fixes * Add changelog fragment * Add support for argcomplete * Enable argcomplete global completion * Rename PrependAction to PrependListAction to better describe what it does * Add documentation for installing and configuring argcomplete * Address rebase issues * Fix display encoding for vault * Fix line length * Address rebase issues * Handle rebase issues * Use mutually exclusive group instead of handling manually * Fix rebase issues * Address rebase issue * Update version added for argcomplete support * -e must be given a value * ci_complete
		
			
				
	
	
		
			184 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright: (c) 2017, Ansible Project
 | |
| # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
 | |
| 
 | |
| from __future__ import (absolute_import, division, print_function)
 | |
| __metaclass__ = type
 | |
| 
 | |
| import os
 | |
| import shlex
 | |
| import subprocess
 | |
| import yaml
 | |
| 
 | |
| from ansible import context
 | |
| from ansible.cli import CLI
 | |
| from ansible.cli.arguments import option_helpers as opt_help
 | |
| from ansible.config.manager import ConfigManager, Setting, find_ini_config_file
 | |
| from ansible.errors import AnsibleError, AnsibleOptionsError
 | |
| from ansible.module_utils._text import to_native, to_text
 | |
| from ansible.parsing.yaml.dumper import AnsibleDumper
 | |
| from ansible.utils.color import stringc
 | |
| from ansible.utils.display import Display
 | |
| from ansible.utils.path import unfrackpath
 | |
| 
 | |
| display = Display()
 | |
| 
 | |
| 
 | |
| class ConfigCLI(CLI):
 | |
|     """ Config command line class """
 | |
| 
 | |
|     def __init__(self, args, callback=None):
 | |
| 
 | |
|         self.config_file = None
 | |
|         self.config = None
 | |
|         super(ConfigCLI, self).__init__(args, callback)
 | |
| 
 | |
|     def init_parser(self):
 | |
| 
 | |
|         super(ConfigCLI, self).init_parser(
 | |
|             desc="View, edit, and manage ansible configuration.",
 | |
|         )
 | |
| 
 | |
|         common = opt_help.argparse.ArgumentParser(add_help=False)
 | |
|         opt_help.add_verbosity_options(common)
 | |
|         common.add_argument('-c', '--config', dest='config_file',
 | |
|                             help="path to configuration file, defaults to first file found in precedence.")
 | |
| 
 | |
|         subparsers = self.parser.add_subparsers(dest='action')
 | |
|         subparsers.required = True
 | |
| 
 | |
|         list_parser = subparsers.add_parser('list', help='Print all config options', parents=[common])
 | |
|         list_parser.set_defaults(func=self.execute_list)
 | |
| 
 | |
|         dump_parser = subparsers.add_parser('dump', help='Dump configuration', parents=[common])
 | |
|         dump_parser.set_defaults(func=self.execute_dump)
 | |
|         dump_parser.add_argument('--only-changed', dest='only_changed', action='store_true',
 | |
|                                  help="Only show configurations that have changed from the default")
 | |
| 
 | |
|         view_parser = subparsers.add_parser('view', help='View configuration file', parents=[common])
 | |
|         view_parser.set_defaults(func=self.execute_view)
 | |
| 
 | |
|         # update_parser = subparsers.add_parser('update', help='Update configuration option')
 | |
|         # update_parser.set_defaults(func=self.execute_update)
 | |
|         # update_parser.add_argument('-s', '--setting', dest='setting',
 | |
|         #                            help="config setting, the section defaults to 'defaults'",
 | |
|         #                            metavar='[section.]setting=value')
 | |
| 
 | |
|         # search_parser = subparsers.add_parser('search', help='Search configuration')
 | |
|         # search_parser.set_defaults(func=self.execute_search)
 | |
|         # search_parser.add_argument('args', help='Search term', metavar='<search term>')
 | |
| 
 | |
|     def post_process_args(self, options):
 | |
|         options = super(ConfigCLI, self).post_process_args(options)
 | |
|         display.verbosity = options.verbosity
 | |
| 
 | |
|         return options
 | |
| 
 | |
|     def run(self):
 | |
| 
 | |
|         super(ConfigCLI, self).run()
 | |
| 
 | |
|         if context.CLIARGS['config_file']:
 | |
|             self.config_file = unfrackpath(context.CLIARGS['config_file'], follow=False)
 | |
|             self.config = ConfigManager(self.config_file)
 | |
|         else:
 | |
|             self.config = ConfigManager()
 | |
|             self.config_file = find_ini_config_file()
 | |
| 
 | |
|         if self.config_file:
 | |
|             try:
 | |
|                 if not os.path.exists(self.config_file):
 | |
|                     raise AnsibleOptionsError("%s does not exist or is not accessible" % (self.config_file))
 | |
|                 elif not os.path.isfile(self.config_file):
 | |
|                     raise AnsibleOptionsError("%s is not a valid file" % (self.config_file))
 | |
| 
 | |
|                 os.environ['ANSIBLE_CONFIG'] = to_native(self.config_file)
 | |
|             except Exception:
 | |
|                 if context.CLIARGS['action'] in ['view']:
 | |
|                     raise
 | |
|                 elif context.CLIARGS['action'] in ['edit', 'update']:
 | |
|                     display.warning("File does not exist, used empty file: %s" % self.config_file)
 | |
| 
 | |
|         elif context.CLIARGS['action'] == 'view':
 | |
|             raise AnsibleError('Invalid or no config file was supplied')
 | |
| 
 | |
|         context.CLIARGS['func']()
 | |
| 
 | |
|     def execute_update(self):
 | |
|         '''
 | |
|         Updates a single setting in the specified ansible.cfg
 | |
|         '''
 | |
|         raise AnsibleError("Option not implemented yet")
 | |
| 
 | |
|         # pylint: disable=unreachable
 | |
|         if context.CLIARGS['setting'] is None:
 | |
|             raise AnsibleOptionsError("update option requires a setting to update")
 | |
| 
 | |
|         (entry, value) = context.CLIARGS['setting'].split('=')
 | |
|         if '.' in entry:
 | |
|             (section, option) = entry.split('.')
 | |
|         else:
 | |
|             section = 'defaults'
 | |
|             option = entry
 | |
|         subprocess.call([
 | |
|             'ansible',
 | |
|             '-m', 'ini_file',
 | |
|             'localhost',
 | |
|             '-c', 'local',
 | |
|             '-a', '"dest=%s section=%s option=%s value=%s backup=yes"' % (self.config_file, section, option, value)
 | |
|         ])
 | |
| 
 | |
|     def execute_view(self):
 | |
|         '''
 | |
|         Displays the current config file
 | |
|         '''
 | |
|         try:
 | |
|             with open(self.config_file, 'rb') as f:
 | |
|                 self.pager(to_text(f.read(), errors='surrogate_or_strict'))
 | |
|         except Exception as e:
 | |
|             raise AnsibleError("Failed to open config file: %s" % to_native(e))
 | |
| 
 | |
|     def execute_edit(self):
 | |
|         '''
 | |
|         Opens ansible.cfg in the default EDITOR
 | |
|         '''
 | |
|         raise AnsibleError("Option not implemented yet")
 | |
| 
 | |
|         # pylint: disable=unreachable
 | |
|         try:
 | |
|             editor = shlex.split(os.environ.get('EDITOR', 'vi'))
 | |
|             editor.append(self.config_file)
 | |
|             subprocess.call(editor)
 | |
|         except Exception as e:
 | |
|             raise AnsibleError("Failed to open editor: %s" % to_native(e))
 | |
| 
 | |
|     def execute_list(self):
 | |
|         '''
 | |
|         list all current configs reading lib/constants.py and shows env and config file setting names
 | |
|         '''
 | |
|         self.pager(to_text(yaml.dump(self.config.get_configuration_definitions(), Dumper=AnsibleDumper), errors='surrogate_or_strict'))
 | |
| 
 | |
|     def execute_dump(self):
 | |
|         '''
 | |
|         Shows the current settings, merges ansible.cfg if specified
 | |
|         '''
 | |
|         # FIXME: deal with plugins, not just base config
 | |
|         text = []
 | |
|         defaults = self.config.get_configuration_definitions().copy()
 | |
|         for setting in self.config.data.get_settings():
 | |
|             if setting.name in defaults:
 | |
|                 defaults[setting.name] = setting
 | |
| 
 | |
|         for setting in sorted(defaults):
 | |
|             if isinstance(defaults[setting], Setting):
 | |
|                 if defaults[setting].origin == 'default':
 | |
|                     color = 'green'
 | |
|                 else:
 | |
|                     color = 'yellow'
 | |
|                 msg = "%s(%s) = %s" % (setting, defaults[setting].origin, defaults[setting].value)
 | |
|             else:
 | |
|                 color = 'green'
 | |
|                 msg = "%s(%s) = %s" % (setting, 'default', defaults[setting].get('default'))
 | |
|             if not context.CLIARGS['only_changed'] or color == 'yellow':
 | |
|                 text.append(stringc(msg, color))
 | |
| 
 | |
|         self.pager(to_text('\n'.join(text), errors='surrogate_or_strict'))
 |