installer_checkpoint.py 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  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. phases = stats.custom['_run']
  17. # Find the longest phase title
  18. max_column = 0
  19. for phase in phases:
  20. max_column = max(max_column, len(phases[phase]['title']))
  21. # Sort the phases by start time
  22. ordered_phases = sorted(phases, key=lambda x: (phases[x]['start']))
  23. self._display.banner('INSTALLER STATUS')
  24. # Display status information for each phase
  25. for phase in ordered_phases:
  26. phase_title = phases[phase]['title']
  27. padding = max_column - len(phase_title) + 2
  28. phase_status = phases[phase]['status']
  29. phase_time = phase_time_delta(phases[phase])
  30. self._display.display(
  31. '{}{}: {} ({})'.format(phase_title, ' ' * padding, phase_status, phase_time),
  32. color=self.phase_color(phase_status))
  33. # If the phase is not complete, tell the user what playbook to rerun
  34. if phase_status == 'In Progress' and phase != 'installer_phase_initialize':
  35. self._display.display(
  36. '\tThis phase can be restarted by running: {}'.format(
  37. phases[phase]['playbook']))
  38. # Display any extra messages stored during the phase
  39. if 'message' in phases[phase]:
  40. self._display.display(
  41. '\t{}'.format(
  42. phases[phase]['message']))
  43. def phase_color(self, status):
  44. """ Return color code for installer phase"""
  45. valid_status = [
  46. 'In Progress',
  47. 'Complete',
  48. ]
  49. if status not in valid_status:
  50. self._display.warning('Invalid phase status defined: {}'.format(status))
  51. if status == 'Complete':
  52. phase_color = C.COLOR_OK
  53. elif status == 'In Progress':
  54. phase_color = C.COLOR_ERROR
  55. else:
  56. phase_color = C.COLOR_WARN
  57. return phase_color
  58. def phase_time_delta(phase):
  59. """ Calculate the difference between phase start and end times """
  60. time_format = '%Y%m%d%H%M%SZ'
  61. phase_start = datetime.strptime(phase['start'], time_format)
  62. if 'end' not in phase:
  63. # The phase failed so set the end time to now
  64. phase_end = datetime.now()
  65. else:
  66. phase_end = datetime.strptime(phase['end'], time_format)
  67. delta = str(phase_end - phase_start).split(".")[0] # Trim microseconds
  68. return delta