ossh 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. #!/usr/bin/env python
  2. # vim: expandtab:tabstop=4:shiftwidth=4
  3. import argparse
  4. import ansibleutil
  5. import traceback
  6. import sys
  7. import os
  8. import re
  9. class Ossh(object):
  10. def __init__(self):
  11. self.file_path = os.path.join(os.path.dirname(os.path.realpath(__file__)))
  12. self.parse_cli_args()
  13. self.ansible = ansibleutil.AnsibleUtil()
  14. # get a dict of host inventory
  15. if self.args.list:
  16. self.get_hosts()
  17. else:
  18. self.get_hosts(True)
  19. # parse host and user
  20. self.process_host()
  21. if self.args.host == '' and not self.args.list:
  22. self.parser.print_help()
  23. return
  24. if self.args.debug:
  25. print self.args
  26. # perform the SSH
  27. if self.args.list:
  28. self.list_hosts()
  29. else:
  30. self.ssh()
  31. def parse_cli_args(self):
  32. parser = argparse.ArgumentParser(description='Openshift Online SSH Tool.')
  33. parser.add_argument('-e', '--env', action="store",
  34. help="Which environment to search for the host ")
  35. parser.add_argument('-d', '--debug', default=False,
  36. action="store_true", help="debug mode")
  37. parser.add_argument('-v', '--verbose', default=False,
  38. action="store_true", help="Verbose?")
  39. parser.add_argument('--list', default=False,
  40. action="store_true", help="list out hosts")
  41. parser.add_argument('-c', '--command', action='store',
  42. help='Command to run on remote host')
  43. parser.add_argument('-l', '--login_name', action='store',
  44. help='User in which to ssh as')
  45. parser.add_argument('-o', '--ssh_opts', action='store',
  46. help='options to pass to SSH.\n \
  47. "-oForwardX11=yes,TCPKeepAlive=yes"')
  48. parser.add_argument('host', nargs='?', default='')
  49. self.args = parser.parse_args()
  50. self.parser = parser
  51. def process_host(self):
  52. '''Determine host name and user name for SSH.
  53. '''
  54. self.env = None
  55. self.user = None
  56. re_env = re.compile("\.(" + "|".join(self.host_inventory.keys()) + ")")
  57. search = re_env.search(self.args.host)
  58. if self.args.env:
  59. self.env = self.args.env
  60. elif search:
  61. # take the first?
  62. self.env = search.groups()[0]
  63. # remove env from hostname command line arg if found
  64. if search:
  65. self.args.host = re_env.split(self.args.host)[0]
  66. # parse username if passed
  67. if '@' in self.args.host:
  68. self.user, self.host = self.args.host.split('@')
  69. else:
  70. self.host = self.args.host
  71. if self.args.login_name:
  72. self.user = self.args.login_name
  73. def get_hosts(self, cache_only=False):
  74. '''Query our host inventory and return a dict where the format
  75. equals:
  76. dict['servername'] = dns_name
  77. '''
  78. if cache_only:
  79. self.host_inventory = self.ansible.build_host_dict_by_env(['--cache-only'])
  80. else:
  81. self.host_inventory = self.ansible.build_host_dict_by_env()
  82. def select_host(self):
  83. '''select host attempts to match the host specified
  84. on the command line with a list of hosts.
  85. '''
  86. results = []
  87. for env in self.host_inventory.keys():
  88. for hostname, server_info in self.host_inventory[env].items():
  89. if hostname.split(':')[0] == self.host:
  90. results.append((hostname, server_info))
  91. # attempt to select the correct environment if specified
  92. if self.env:
  93. results = filter(lambda result: result[1]['ec2_tag_environment'] == self.env, results)
  94. if results:
  95. return results
  96. else:
  97. print "Could not find specified host: %s." % self.host
  98. # default - no results found.
  99. return None
  100. def list_hosts(self, limit=None):
  101. '''Function to print out the host inventory.
  102. Takes a single parameter to limit the number of hosts printed.
  103. '''
  104. if self.env:
  105. results = self.select_host()
  106. if len(results) == 1:
  107. hostname, server_info = results[0]
  108. sorted_keys = server_info.keys()
  109. sorted_keys.sort()
  110. for key in sorted_keys:
  111. print '{0:<35} {1}'.format(key, server_info[key])
  112. else:
  113. for host_id, server_info in results[:limit]:
  114. name = server_info['ec2_tag_Name']
  115. ec2_id = server_info['ec2_id']
  116. ip = server_info['ec2_ip_address']
  117. print '{ec2_tag_Name:<35} {ec2_tag_environment:<8} {ec2_id:<15} {ec2_ip_address}'.format(**server_info)
  118. if limit:
  119. print
  120. print 'Showing only the first %d results...' % limit
  121. print
  122. else:
  123. for env, host_ids in self.host_inventory.items():
  124. for host_id, server_info in host_ids.items():
  125. name = server_info['ec2_tag_Name']
  126. ec2_id = server_info['ec2_id']
  127. ip = server_info['ec2_ip_address']
  128. print '{ec2_tag_Name:<35} {ec2_tag_environment:<5} {ec2_id:<15} {ec2_ip_address}'.format(**server_info)
  129. def ssh(self):
  130. '''SSH to a specified host
  131. '''
  132. try:
  133. # shell args start with the program name in position 1
  134. ssh_args = ['/usr/bin/ssh']
  135. if self.user:
  136. ssh_args.append('-l%s' % self.user)
  137. if self.args.verbose:
  138. ssh_args.append('-vvv')
  139. if self.args.ssh_opts:
  140. for arg in self.args.ssh_opts.split(","):
  141. ssh_args.append("-o%s" % arg)
  142. results = self.select_host()
  143. if not results:
  144. return # early exit, no results
  145. if len(results) > 1:
  146. print "Multiple results found for %s." % self.host
  147. for result in results:
  148. print "{ec2_tag_Name:<35} {ec2_tag_environment:<5} {ec2_id:<10}".format(**result[1])
  149. return # early exit, too many results
  150. # Assume we have one and only one.
  151. hostname, server_info = results[0]
  152. dns = server_info['ec2_public_dns_name']
  153. ssh_args.append(dns)
  154. #last argument
  155. if self.args.command:
  156. ssh_args.append("%s" % self.args.command)
  157. print "Running: %s\n" % ' '.join(ssh_args)
  158. os.execve('/usr/bin/ssh', ssh_args, os.environ)
  159. except:
  160. print traceback.print_exc()
  161. print sys.exc_info()
  162. if __name__ == '__main__':
  163. ossh = Ossh()