opssh 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. #!/usr/bin/env python
  2. # vim: expandtab:tabstop=4:shiftwidth=4
  3. import argparse
  4. import traceback
  5. import sys
  6. import os
  7. import re
  8. import tempfile
  9. import time
  10. import subprocess
  11. import ConfigParser
  12. from openshift_ansible import awsutil
  13. from openshift_ansible.awsutil import ArgumentError
  14. DEFAULT_PSSH_PAR = 200
  15. PSSH = '/usr/bin/pssh'
  16. CONFIG_MAIN_SECTION = 'main'
  17. CONFIG_HOST_TYPE_ALIAS_SECTION = 'host_type_aliases'
  18. CONFIG_INVENTORY_OPTION = 'inventory'
  19. class Opssh(object):
  20. def __init__(self):
  21. self.inventory = None
  22. self.host_type_aliases = {}
  23. self.file_path = os.path.join(os.path.dirname(os.path.realpath(__file__)))
  24. # Default the config path to /etc
  25. self.config_path = os.path.join(os.path.sep, 'etc', \
  26. 'openshift_ansible', \
  27. 'openshift_ansible.conf')
  28. self.parse_cli_args()
  29. self.parse_config_file()
  30. self.aws = awsutil.AwsUtil(self.inventory, self.host_type_aliases)
  31. def run(self):
  32. if self.args.list_host_types:
  33. self.aws.print_host_types()
  34. return 0
  35. if self.args.host_type is not None or \
  36. self.args.env is not None:
  37. return self.run_pssh()
  38. # We weren't able to determine what they wanted to do
  39. raise ArgumentError("Invalid combination of arguments")
  40. def run_pssh(self):
  41. """Actually run the pssh command based off of the supplied options
  42. """
  43. # Default set of options
  44. pssh_args = [PSSH, '-t', '0', '-p', str(self.args.par), '--user', self.args.user]
  45. if self.args.inline:
  46. pssh_args.append("--inline")
  47. if self.args.outdir:
  48. pssh_args.extend(["--outdir", self.args.outdir])
  49. if self.args.errdir:
  50. pssh_args.extend(["--errdir", self.args.errdir])
  51. hosts = self.aws.get_host_list(host_type=self.args.host_type,
  52. env=self.args.env)
  53. with tempfile.NamedTemporaryFile(prefix='opssh-', delete=True) as f:
  54. for h in hosts:
  55. f.write(h + os.linesep)
  56. f.flush()
  57. pssh_args.extend(["-h", f.name])
  58. pssh_args.append(self.args.command)
  59. print
  60. print "Running: %s" % ' '.join(pssh_args)
  61. print
  62. return subprocess.call(pssh_args)
  63. return None
  64. def parse_config_file(self):
  65. if os.path.isfile(self.config_path):
  66. config = ConfigParser.ConfigParser()
  67. config.read(self.config_path)
  68. if config.has_section(CONFIG_MAIN_SECTION) and \
  69. config.has_option(CONFIG_MAIN_SECTION, CONFIG_INVENTORY_OPTION):
  70. self.inventory = config.get(CONFIG_MAIN_SECTION, CONFIG_INVENTORY_OPTION)
  71. self.host_type_aliases = {}
  72. if config.has_section(CONFIG_HOST_TYPE_ALIAS_SECTION):
  73. for alias in config.options(CONFIG_HOST_TYPE_ALIAS_SECTION):
  74. value = config.get(CONFIG_HOST_TYPE_ALIAS_SECTION, alias).split(',')
  75. self.host_type_aliases[alias] = value
  76. def parse_cli_args(self):
  77. """Setup the command line parser with the options we want
  78. """
  79. parser = argparse.ArgumentParser(description='Openshift Online PSSH Tool.')
  80. parser.add_argument('--list-host-types', default=False, action='store_true',
  81. help='List all of the host types')
  82. parser.add_argument('-e', '--env', action="store",
  83. help="Which environment to use")
  84. parser.add_argument('-t', '--host-type', action="store", default=None,
  85. help="Which host type to use")
  86. parser.add_argument('-c', '--command', action='store',
  87. help='Command to run on remote host(s)')
  88. parser.add_argument('--user', action='store', default='root',
  89. help='username')
  90. parser.add_argument('-i', '--inline', default=False, action='store_true',
  91. help='inline aggregated output and error for each server')
  92. parser.add_argument('-p', '--par', action='store', default=DEFAULT_PSSH_PAR,
  93. help=('max number of parallel threads (default %s)' % DEFAULT_PSSH_PAR))
  94. parser.add_argument('--outdir', action='store',
  95. help='output directory for stdout files')
  96. parser.add_argument('--errdir', action='store',
  97. help='output directory for stderr files')
  98. self.args = parser.parse_args()
  99. if __name__ == '__main__':
  100. if len(sys.argv) == 1:
  101. print "\nError: No options given. Use --help to see the available options\n"
  102. sys.exit(0)
  103. try:
  104. opssh = Opssh()
  105. exitcode = opssh.run()
  106. sys.exit(exitcode)
  107. except ArgumentError as e:
  108. print "\nError: %s\n" % e.message