/usr/lib/python3/dist-packages/tomahawk/command.py is in python3-tomahawk 0.7.1-2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 | # -*- coding: utf-8 -*-
import argparse
import getpass
import os
import signal
import sys
import time
from tomahawk.base import BaseContext, BaseExecutor, BaseMain
from tomahawk.color import (
create_coloring_object
)
from tomahawk.constants import (
DEFAULT_COMMAND_OUTPUT_FORMAT
)
from tomahawk.expect import CommandWithExpect
from tomahawk.utils import (
shutdown_by_signal,
check_required_command
)
class CommandContext(BaseContext):
"""
Command context
"""
def __init__(self, arguments, options, out, err):
super(CommandContext, self).__init__(options, out, err)
self.arguments = arguments
class CommandMain(BaseMain):
"""
Main class for tomahawk
"""
def __init__(self, file):
# setup self.log, self.arg_parser, self.options
super(CommandMain, self).__init__(file)
self.log.debug("options = " + str(self.options))
self.log.debug("arguments = " + str(self.options.command))
def do_run(self):
self.context = CommandContext(
self.options.command,
self.options.__dict__,
sys.stdout,
sys.stderr
)
check_required_command('ssh')
hosts = self.check_hosts()
color = create_coloring_object(sys.stdout)
# prompt when production environment
self.confirm_execution_on_production(
'Command "%s" will be executed to %s hosts. Are you sure? [yes/NO]: '
% (color.green(' '.join(self.context.arguments)), color.green(len(hosts)))
)
executor = CommandExecutor(self.context, self.log, hosts)
return executor.execute(self.context.arguments)
@classmethod
def create_argument_parser(cls, file):
parser = argparse.ArgumentParser(
prog = os.path.basename(file),
description = "A simple command executor for many hosts.",
conflict_handler = 'resolve'
)
parser.add_argument(
'command', metavar='command', nargs='*',
help='Command executed on remote hosts.',
)
parser.add_argument(
'--ssh', default='ssh', help='ssh program. (default: "ssh")'
)
parser.add_argument(
'-u', '--ssh-user', help='ssh user.'
)
parser.add_argument(
'-o', '--ssh-options', help='ssh options.'
)
parser.add_argument(
'-s', '--prompt-sudo-password', action='store_true',
help='Prompt a password for sudo.'
)
parser.add_argument(
'--sudo-password-stdin', action='store_true',
help='Read a password for sudo from stdin.'
)
parser.add_argument(
'-F', '--output-format', default=DEFAULT_COMMAND_OUTPUT_FORMAT,
help="Command output format. (default: '%s')" % (DEFAULT_COMMAND_OUTPUT_FORMAT.replace('%', '%%').replace('\n', '\\n'))
)
parser.add_argument(
'-V', '--verify-output', action='store_true',
help="Verify command output of all hosts."
)
cls.add_common_arguments(parser)
return parser
def _command(
command, command_args, login_password, sudo_password,
timeout, expect_delay, debug_enabled):
"""
Execute a command.
"""
# Trap SIGINT(Ctrl-C) to quit executing a command
signal.signal(signal.SIGINT, shutdown_by_signal)
try:
return CommandWithExpect(
command, command_args, login_password, sudo_password,
timeout, expect_delay, debug_enabled
).execute()
except:
from traceback import print_tb
print("""%s: %s""" % (sys.exc_info()[0], sys.exc_info()[1]))
print_tb(sys.exc_info()[2])
raise
class CommandExecutor(BaseExecutor):
"""
Execute commands.
Args:
commands -- commands to execute.
Returns: when rsync succeeds, return 0. When errors, return 1
"""
def execute(self, commands):
if len(commands) == 0:
raise RuntimeError("[error] Too few arguments")
options = self.context.options
#ssh = options.get('ssh') or 'ssh'
ssh_user = options.get('ssh_user') or ''
ssh_options = ''
if options.get('ssh_options'):
ssh_options = options['ssh_options'] + ' '
if ssh_user:
ssh_options += '-l ' + ssh_user
if ssh_options.find('-T') == -1:
# if '-T' isn't specified, turn 'pseudo-tty allocation' on
ssh_options += ' -t'
async_results = []
for host in self.hosts:
for command in commands:
command_args = []
for option in ssh_options.split(' '):
# remove left and right whitespaces
option = option.strip()
if len(option) > 0:
command_args.append(option)
command_args.append(host)
# Escape shell special chars
c = command.replace('\\', '\\\\') \
.replace('"', '\"') \
.replace('$', '\$') \
.replace('`', '\`')
# execute a command with shell because we want to use pipe(|) and so on.
command_args.extend([ '/bin/sh', '-c', '"%s"' % (c) ])
# host, command, ssh_user, ssh_option, login_password, sudo_password
async_result = self.process_pool.apply_async(
_command,
( 'ssh', command_args, self.login_password, self.sudo_password,
options['timeout'], options['expect_delay'], options['debug'] ),
)
async_results.append({ 'host': host, 'command': command, 'async_result': async_result })
if options['delay'] != 0:
time.sleep(options['delay'])
#######################
# callbacks
#######################
def create_output(color, output_format_template, command, host, exit_status, command_output):
c = command
if exit_status == 0:
c = color.green(command)
return output_format_template.safe_substitute({
'user': ssh_user or '[user]',
'host': host,
'command': c,
'output': command_output,
})
def create_timeout_message(color, output, timeout):
output += 'Command timed out after %d seconds' % (options['timeout'])
return output
def create_timeout_raise_error_message(color, command, host, timeout):
return 'Command "%s" timed out on host "%s" after %d seconds' \
% (command, host, timeout)
def create_failure_message(color, output, exit_status):
output += 'Command failed ! (status = %d)' % exit_status
return output
def create_failure_raise_error_message(color, command, host):
return 'Command "%s" failed on host "%s"' % (command, host)
def create_failure_last_message(color, command, hosts):
return 'Command "%s" failed on following hosts\n%s' % (command, hosts)
# Call BaseExectuor#process_async_results with callbacks
return self.process_async_results(
async_results,
create_output,
create_timeout_message,
create_timeout_raise_error_message,
create_failure_message,
create_failure_raise_error_message,
create_failure_last_message
)
|