installer_checkpoint.py 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. """Ansible callback plugin to print a summary completion status of installation
  2. phases.
  3. """
  4. from datetime import datetime
  5. from ansible.plugins.callback import CallbackBase
  6. from ansible import constants as C
  7. class CallbackModule(CallbackBase):
  8. """This callback summarizes installation phase status."""
  9. CALLBACK_VERSION = 2.0
  10. CALLBACK_TYPE = 'aggregate'
  11. CALLBACK_NAME = 'installer_checkpoint'
  12. CALLBACK_NEEDS_WHITELIST = False
  13. def __init__(self):
  14. super(CallbackModule, self).__init__()
  15. def v2_playbook_on_stats(self, stats):
  16. # Set the order of the installer phases
  17. installer_phases = [
  18. 'installer_phase_initialize',
  19. 'installer_phase_health',
  20. 'installer_phase_etcd',
  21. 'installer_phase_nfs',
  22. 'installer_phase_loadbalancer',
  23. 'installer_phase_master',
  24. 'installer_phase_master_additional',
  25. 'installer_phase_node',
  26. 'installer_phase_glusterfs',
  27. 'installer_phase_hosted',
  28. 'installer_phase_web_console',
  29. 'installer_phase_metrics',
  30. 'installer_phase_logging',
  31. 'installer_phase_prometheus',
  32. 'installer_phase_servicecatalog',
  33. 'installer_phase_management',
  34. ]
  35. # Define the attributes of the installer phases
  36. phase_attributes = {
  37. 'installer_phase_initialize': {
  38. 'title': 'Initialization',
  39. 'playbook': ''
  40. },
  41. 'installer_phase_health': {
  42. 'title': 'Health Check',
  43. 'playbook': 'playbooks/openshift-checks/pre-install.yml'
  44. },
  45. 'installer_phase_etcd': {
  46. 'title': 'etcd Install',
  47. 'playbook': 'playbooks/openshift-etcd/config.yml'
  48. },
  49. 'installer_phase_nfs': {
  50. 'title': 'NFS Install',
  51. 'playbook': 'playbooks/openshift-nfs/config.yml'
  52. },
  53. 'installer_phase_loadbalancer': {
  54. 'title': 'Load balancer Install',
  55. 'playbook': 'playbooks/openshift-loadbalancer/config.yml'
  56. },
  57. 'installer_phase_master': {
  58. 'title': 'Master Install',
  59. 'playbook': 'playbooks/openshift-master/config.yml'
  60. },
  61. 'installer_phase_master_additional': {
  62. 'title': 'Master Additional Install',
  63. 'playbook': 'playbooks/openshift-master/additional_config.yml'
  64. },
  65. 'installer_phase_node': {
  66. 'title': 'Node Install',
  67. 'playbook': 'playbooks/openshift-node/config.yml'
  68. },
  69. 'installer_phase_glusterfs': {
  70. 'title': 'GlusterFS Install',
  71. 'playbook': 'playbooks/openshift-glusterfs/config.yml'
  72. },
  73. 'installer_phase_hosted': {
  74. 'title': 'Hosted Install',
  75. 'playbook': 'playbooks/openshift-hosted/config.yml'
  76. },
  77. 'installer_phase_web_console': {
  78. 'title': 'Web Console Install',
  79. 'playbook': 'playbooks/openshift-web-console/config.yml'
  80. },
  81. 'installer_phase_metrics': {
  82. 'title': 'Metrics Install',
  83. 'playbook': 'playbooks/openshift-metrics/config.yml'
  84. },
  85. 'installer_phase_logging': {
  86. 'title': 'Logging Install',
  87. 'playbook': 'playbooks/openshift-logging/config.yml'
  88. },
  89. 'installer_phase_prometheus': {
  90. 'title': 'Prometheus Install',
  91. 'playbook': 'playbooks/openshift-prometheus/config.yml'
  92. },
  93. 'installer_phase_servicecatalog': {
  94. 'title': 'Service Catalog Install',
  95. 'playbook': 'playbooks/openshift-service-catalog/config.yml'
  96. },
  97. 'installer_phase_management': {
  98. 'title': 'Management Install',
  99. 'playbook': 'playbooks/openshift-management/config.yml'
  100. },
  101. }
  102. # Find the longest phase title
  103. max_column = 0
  104. for phase in phase_attributes:
  105. max_column = max(max_column, len(phase_attributes[phase]['title']))
  106. if '_run' in stats.custom:
  107. self._display.banner('INSTALLER STATUS')
  108. for phase in installer_phases:
  109. phase_title = phase_attributes[phase]['title']
  110. padding = max_column - len(phase_title) + 2
  111. if phase in stats.custom['_run']:
  112. phase_status = stats.custom['_run'][phase]['status']
  113. phase_time = phase_time_delta(stats.custom['_run'][phase])
  114. self._display.display(
  115. '{}{}: {} ({})'.format(phase_title, ' ' * padding, phase_status, phase_time),
  116. color=self.phase_color(phase_status))
  117. if phase_status == 'In Progress' and phase != 'installer_phase_initialize':
  118. self._display.display(
  119. '\tThis phase can be restarted by running: {}'.format(
  120. phase_attributes[phase]['playbook']))
  121. if 'message' in stats.custom['_run'][phase]:
  122. self._display.display(
  123. '\t{}'.format(
  124. stats.custom['_run'][phase]['message']))
  125. self._display.display("", screen_only=True)
  126. def phase_color(self, status):
  127. """ Return color code for installer phase"""
  128. valid_status = [
  129. 'In Progress',
  130. 'Complete',
  131. ]
  132. if status not in valid_status:
  133. self._display.warning('Invalid phase status defined: {}'.format(status))
  134. if status == 'Complete':
  135. phase_color = C.COLOR_OK
  136. elif status == 'In Progress':
  137. phase_color = C.COLOR_ERROR
  138. else:
  139. phase_color = C.COLOR_WARN
  140. return phase_color
  141. def phase_time_delta(phase):
  142. """ Calculate the difference between phase start and end times """
  143. time_format = '%Y%m%d%H%M%SZ'
  144. phase_start = datetime.strptime(phase['start'], time_format)
  145. if 'end' not in phase:
  146. # The phase failed so set the end time to now
  147. phase_end = datetime.now()
  148. else:
  149. phase_end = datetime.strptime(phase['end'], time_format)
  150. delta = str(phase_end - phase_start).split(".")[0] # Trim microseconds
  151. return delta