installer_checkpoint.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  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_metrics',
  29. 'installer_phase_logging',
  30. 'installer_phase_prometheus',
  31. 'installer_phase_servicecatalog',
  32. 'installer_phase_management',
  33. ]
  34. # Define the attributes of the installer phases
  35. phase_attributes = {
  36. 'installer_phase_initialize': {
  37. 'title': 'Initialization',
  38. 'playbook': ''
  39. },
  40. 'installer_phase_health': {
  41. 'title': 'Health Check',
  42. 'playbook': 'playbooks/openshift-checks/pre-install.yml'
  43. },
  44. 'installer_phase_etcd': {
  45. 'title': 'etcd Install',
  46. 'playbook': 'playbooks/openshift-etcd/config.yml'
  47. },
  48. 'installer_phase_nfs': {
  49. 'title': 'NFS Install',
  50. 'playbook': 'playbooks/openshift-nfs/config.yml'
  51. },
  52. 'installer_phase_loadbalancer': {
  53. 'title': 'Load balancer Install',
  54. 'playbook': 'playbooks/openshift-loadbalancer/config.yml'
  55. },
  56. 'installer_phase_master': {
  57. 'title': 'Master Install',
  58. 'playbook': 'playbooks/openshift-master/config.yml'
  59. },
  60. 'installer_phase_master_additional': {
  61. 'title': 'Master Additional Install',
  62. 'playbook': 'playbooks/openshift-master/additional_config.yml'
  63. },
  64. 'installer_phase_node': {
  65. 'title': 'Node Install',
  66. 'playbook': 'playbooks/openshift-node/config.yml'
  67. },
  68. 'installer_phase_glusterfs': {
  69. 'title': 'GlusterFS Install',
  70. 'playbook': 'playbooks/byo/openshift-glusterfs/config.yml'
  71. },
  72. 'installer_phase_hosted': {
  73. 'title': 'Hosted Install',
  74. 'playbook': 'playbooks/openshift-hosted/config.yml'
  75. },
  76. 'installer_phase_metrics': {
  77. 'title': 'Metrics Install',
  78. 'playbook': 'playbooks/openshift-metrics/config.yml'
  79. },
  80. 'installer_phase_logging': {
  81. 'title': 'Logging Install',
  82. 'playbook': 'playbooks/byo/openshift-cluster/openshift-logging.yml'
  83. },
  84. 'installer_phase_prometheus': {
  85. 'title': 'Prometheus Install',
  86. 'playbook': 'playbooks/openshift-prometheus/config.yml'
  87. },
  88. 'installer_phase_servicecatalog': {
  89. 'title': 'Service Catalog Install',
  90. 'playbook': 'playbooks/openshift-service-catalog/config.yml'
  91. },
  92. 'installer_phase_management': {
  93. 'title': 'Management Install',
  94. 'playbook': 'playbooks/openshift-management/config.yml'
  95. },
  96. }
  97. # Find the longest phase title
  98. max_column = 0
  99. for phase in phase_attributes:
  100. max_column = max(max_column, len(phase_attributes[phase]['title']))
  101. if '_run' in stats.custom:
  102. self._display.banner('INSTALLER STATUS')
  103. for phase in installer_phases:
  104. phase_title = phase_attributes[phase]['title']
  105. padding = max_column - len(phase_title) + 2
  106. if phase in stats.custom['_run']:
  107. phase_status = stats.custom['_run'][phase]['status']
  108. phase_time = phase_time_delta(stats.custom['_run'][phase])
  109. self._display.display(
  110. '{}{}: {} ({})'.format(phase_title, ' ' * padding, phase_status, phase_time),
  111. color=self.phase_color(phase_status))
  112. if phase_status == 'In Progress' and phase != 'installer_phase_initialize':
  113. self._display.display(
  114. '\tThis phase can be restarted by running: {}'.format(
  115. phase_attributes[phase]['playbook']))
  116. self._display.display("", screen_only=True)
  117. def phase_color(self, status):
  118. """ Return color code for installer phase"""
  119. valid_status = [
  120. 'In Progress',
  121. 'Complete',
  122. ]
  123. if status not in valid_status:
  124. self._display.warning('Invalid phase status defined: {}'.format(status))
  125. if status == 'Complete':
  126. phase_color = C.COLOR_OK
  127. elif status == 'In Progress':
  128. phase_color = C.COLOR_ERROR
  129. else:
  130. phase_color = C.COLOR_WARN
  131. return phase_color
  132. def phase_time_delta(phase):
  133. """ Calculate the difference between phase start and end times """
  134. time_format = '%Y%m%d%H%M%SZ'
  135. phase_start = datetime.strptime(phase['start'], time_format)
  136. if 'end' not in phase:
  137. # The phase failed so set the end time to now
  138. phase_end = datetime.now()
  139. else:
  140. phase_end = datetime.strptime(phase['end'], time_format)
  141. delta = str(phase_end - phase_start).split(".")[0] # Trim microseconds
  142. return delta