9ff03e6c1e
Some downstreams want to ship multiple versions of ansible (Either to have multiple ansible versions or to have a version that uses python3.X and a version that uses python2.x). When they do this, they append a version number to the cli scripts in /usr/bin. This patch will remove those version numbers before trying to find the ansible python module to import for this commandline
145 lines
4.8 KiB
Python
Executable file
145 lines
4.8 KiB
Python
Executable file
#!/usr/bin/env python
|
|
|
|
# (c) 2012, Michael DeHaan <michael.dehaan@gmail.com>
|
|
#
|
|
# This file is part of Ansible
|
|
#
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
########################################################
|
|
from __future__ import (absolute_import, division, print_function)
|
|
__metaclass__ = type
|
|
|
|
__requires__ = ['ansible']
|
|
try:
|
|
import pkg_resources
|
|
except Exception:
|
|
# Use pkg_resources to find the correct versions of libraries and set
|
|
# sys.path appropriately when there are multiversion installs. But we
|
|
# have code that better expresses the errors in the places where the code
|
|
# is actually used (the deps are optional for many code paths) so we don't
|
|
# want to fail here.
|
|
pass
|
|
|
|
import os
|
|
import shutil
|
|
import sys
|
|
import traceback
|
|
|
|
# for debug
|
|
from multiprocessing import Lock
|
|
|
|
import ansible.constants as C
|
|
from ansible.errors import AnsibleError, AnsibleOptionsError, AnsibleParserError
|
|
from ansible.utils.display import Display
|
|
from ansible.module_utils._text import to_text
|
|
|
|
|
|
########################################
|
|
### OUTPUT OF LAST RESORT ###
|
|
class LastResort(object):
|
|
def display(self, msg):
|
|
print(msg, file=sys.stderr)
|
|
|
|
def error(self, msg, wrap_text=None):
|
|
print(msg, file=sys.stderr)
|
|
|
|
|
|
########################################
|
|
|
|
if __name__ == '__main__':
|
|
|
|
display = LastResort()
|
|
cli = None
|
|
me = os.path.basename(sys.argv[0])
|
|
|
|
try:
|
|
display = Display()
|
|
display.debug("starting run")
|
|
|
|
sub = None
|
|
target = me.split('-')
|
|
if target[-1][0].isdigit():
|
|
# Remove any version or pthon version info as downstreams
|
|
# sometimes add that
|
|
target = target[:-1]
|
|
|
|
if len(target) > 1:
|
|
sub = target[1]
|
|
myclass = "%sCLI" % sub.capitalize()
|
|
elif target[0] == 'ansible':
|
|
sub = 'adhoc'
|
|
myclass = 'AdHocCLI'
|
|
else:
|
|
raise AnsibleError("Unknown Ansible alias: %s" % me)
|
|
|
|
try:
|
|
mycli = getattr(__import__("ansible.cli.%s" % sub, fromlist=[myclass]), myclass)
|
|
except ImportError as e:
|
|
# ImportError members have changed in py3
|
|
if 'msg' in dir(e):
|
|
msg = e.msg
|
|
else:
|
|
msg = e.message
|
|
if msg.endswith(' %s' % sub):
|
|
raise AnsibleError("Ansible sub-program not implemented: %s" % me)
|
|
else:
|
|
raise
|
|
|
|
try:
|
|
args = [to_text(a, errors='surrogate_or_strict') for a in sys.argv]
|
|
except UnicodeError:
|
|
display.error('Command line args are not in utf-8, unable to continue. Ansible currently only understands utf-8')
|
|
display.display(u"The full traceback was:\n\n%s" % to_text(traceback.format_exc()))
|
|
exit_code = 6
|
|
else:
|
|
cli = mycli(args)
|
|
cli.parse()
|
|
exit_code = cli.run()
|
|
|
|
except AnsibleOptionsError as e:
|
|
cli.parser.print_help()
|
|
display.error(to_text(e), wrap_text=False)
|
|
exit_code = 5
|
|
except AnsibleParserError as e:
|
|
display.error(to_text(e), wrap_text=False)
|
|
exit_code = 4
|
|
# TQM takes care of these, but leaving comment to reserve the exit codes
|
|
# except AnsibleHostUnreachable as e:
|
|
# display.error(str(e))
|
|
# exit_code = 3
|
|
# except AnsibleHostFailed as e:
|
|
# display.error(str(e))
|
|
# exit_code = 2
|
|
except AnsibleError as e:
|
|
display.error(to_text(e), wrap_text=False)
|
|
exit_code = 1
|
|
except KeyboardInterrupt:
|
|
display.error("User interrupted execution")
|
|
exit_code = 99
|
|
except Exception as e:
|
|
have_cli_options = cli is not None and cli.options is not None
|
|
display.error("Unexpected Exception: %s" % to_text(e), wrap_text=False)
|
|
if not have_cli_options or have_cli_options and cli.options.verbosity > 2:
|
|
log_only = False
|
|
else:
|
|
display.display("to see the full traceback, use -vvv")
|
|
log_only = True
|
|
display.display(u"the full traceback was:\n\n%s" % to_text(traceback.format_exc()), log_only=log_only)
|
|
exit_code = 250
|
|
finally:
|
|
# Remove ansible tempdir
|
|
shutil.rmtree(C.DEFAULT_LOCAL_TMP, True)
|
|
|
|
sys.exit(exit_code)
|