test_git_trace2_event_log.py 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. # Copyright (C) 2020 The Android Open Source Project
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License");
  4. # you may not use this file except in compliance with the License.
  5. # You may obtain a copy of the License at
  6. #
  7. # http://www.apache.org/licenses/LICENSE-2.0
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. """Unittests for the git_trace2_event_log.py module."""
  15. import json
  16. import os
  17. import tempfile
  18. import unittest
  19. from unittest import mock
  20. import git_trace2_event_log
  21. class EventLogTestCase(unittest.TestCase):
  22. """TestCase for the EventLog module."""
  23. PARENT_SID_KEY = 'GIT_TRACE2_PARENT_SID'
  24. PARENT_SID_VALUE = 'parent_sid'
  25. SELF_SID_REGEX = r'repo-\d+T\d+Z-.*'
  26. FULL_SID_REGEX = r'^%s/%s' % (PARENT_SID_VALUE, SELF_SID_REGEX)
  27. def setUp(self):
  28. """Load the event_log module every time."""
  29. self._event_log_module = None
  30. # By default we initialize with the expected case where
  31. # repo launches us (so GIT_TRACE2_PARENT_SID is set).
  32. env = {
  33. self.PARENT_SID_KEY: self.PARENT_SID_VALUE,
  34. }
  35. self._event_log_module = git_trace2_event_log.EventLog(env=env)
  36. self._log_data = None
  37. def verifyCommonKeys(self, log_entry, expected_event_name, full_sid=True):
  38. """Helper function to verify common event log keys."""
  39. self.assertIn('event', log_entry)
  40. self.assertIn('sid', log_entry)
  41. self.assertIn('thread', log_entry)
  42. self.assertIn('time', log_entry)
  43. # Do basic data format validation.
  44. self.assertEqual(expected_event_name, log_entry['event'])
  45. if full_sid:
  46. self.assertRegex(log_entry['sid'], self.FULL_SID_REGEX)
  47. else:
  48. self.assertRegex(log_entry['sid'], self.SELF_SID_REGEX)
  49. self.assertRegex(log_entry['time'], r'^\d+-\d+-\d+T\d+:\d+:\d+\.\d+Z$')
  50. def readLog(self, log_path):
  51. """Helper function to read log data into a list."""
  52. log_data = []
  53. with open(log_path, mode='rb') as f:
  54. for line in f:
  55. log_data.append(json.loads(line))
  56. return log_data
  57. def test_initial_state_with_parent_sid(self):
  58. """Test initial state when 'GIT_TRACE2_PARENT_SID' is set by parent."""
  59. self.assertRegex(self._event_log_module.full_sid, self.FULL_SID_REGEX)
  60. def test_initial_state_no_parent_sid(self):
  61. """Test initial state when 'GIT_TRACE2_PARENT_SID' is not set."""
  62. # Setup an empty environment dict (no parent sid).
  63. self._event_log_module = git_trace2_event_log.EventLog(env={})
  64. self.assertRegex(self._event_log_module.full_sid, self.SELF_SID_REGEX)
  65. def test_version_event(self):
  66. """Test 'version' event data is valid.
  67. Verify that the 'version' event is written even when no other
  68. events are addded.
  69. Expected event log:
  70. <version event>
  71. """
  72. with tempfile.TemporaryDirectory(prefix='event_log_tests') as tempdir:
  73. log_path = self._event_log_module.Write(path=tempdir)
  74. self._log_data = self.readLog(log_path)
  75. # A log with no added events should only have the version entry.
  76. self.assertEqual(len(self._log_data), 1)
  77. version_event = self._log_data[0]
  78. self.verifyCommonKeys(version_event, expected_event_name='version')
  79. # Check for 'version' event specific fields.
  80. self.assertIn('evt', version_event)
  81. self.assertIn('exe', version_event)
  82. # Verify "evt" version field is a string.
  83. self.assertIsInstance(version_event['evt'], str)
  84. def test_start_event(self):
  85. """Test and validate 'start' event data is valid.
  86. Expected event log:
  87. <version event>
  88. <start event>
  89. """
  90. self._event_log_module.StartEvent()
  91. with tempfile.TemporaryDirectory(prefix='event_log_tests') as tempdir:
  92. log_path = self._event_log_module.Write(path=tempdir)
  93. self._log_data = self.readLog(log_path)
  94. self.assertEqual(len(self._log_data), 2)
  95. start_event = self._log_data[1]
  96. self.verifyCommonKeys(self._log_data[0], expected_event_name='version')
  97. self.verifyCommonKeys(start_event, expected_event_name='start')
  98. # Check for 'start' event specific fields.
  99. self.assertIn('argv', start_event)
  100. self.assertTrue(isinstance(start_event['argv'], list))
  101. def test_exit_event_result_none(self):
  102. """Test 'exit' event data is valid when result is None.
  103. We expect None result to be converted to 0 in the exit event data.
  104. Expected event log:
  105. <version event>
  106. <exit event>
  107. """
  108. self._event_log_module.ExitEvent(None)
  109. with tempfile.TemporaryDirectory(prefix='event_log_tests') as tempdir:
  110. log_path = self._event_log_module.Write(path=tempdir)
  111. self._log_data = self.readLog(log_path)
  112. self.assertEqual(len(self._log_data), 2)
  113. exit_event = self._log_data[1]
  114. self.verifyCommonKeys(self._log_data[0], expected_event_name='version')
  115. self.verifyCommonKeys(exit_event, expected_event_name='exit')
  116. # Check for 'exit' event specific fields.
  117. self.assertIn('code', exit_event)
  118. # 'None' result should convert to 0 (successful) return code.
  119. self.assertEqual(exit_event['code'], 0)
  120. def test_exit_event_result_integer(self):
  121. """Test 'exit' event data is valid when result is an integer.
  122. Expected event log:
  123. <version event>
  124. <exit event>
  125. """
  126. self._event_log_module.ExitEvent(2)
  127. with tempfile.TemporaryDirectory(prefix='event_log_tests') as tempdir:
  128. log_path = self._event_log_module.Write(path=tempdir)
  129. self._log_data = self.readLog(log_path)
  130. self.assertEqual(len(self._log_data), 2)
  131. exit_event = self._log_data[1]
  132. self.verifyCommonKeys(self._log_data[0], expected_event_name='version')
  133. self.verifyCommonKeys(exit_event, expected_event_name='exit')
  134. # Check for 'exit' event specific fields.
  135. self.assertIn('code', exit_event)
  136. self.assertEqual(exit_event['code'], 2)
  137. def test_write_with_filename(self):
  138. """Test Write() with a path to a file exits with None."""
  139. self.assertIsNone(self._event_log_module.Write(path='path/to/file'))
  140. def test_write_with_git_config(self):
  141. """Test Write() uses the git config path when 'git config' call succeeds."""
  142. with tempfile.TemporaryDirectory(prefix='event_log_tests') as tempdir:
  143. with mock.patch.object(self._event_log_module,
  144. '_GetEventTargetPath', return_value=tempdir):
  145. self.assertEqual(os.path.dirname(self._event_log_module.Write()), tempdir)
  146. def test_write_no_git_config(self):
  147. """Test Write() with no git config variable present exits with None."""
  148. with mock.patch.object(self._event_log_module,
  149. '_GetEventTargetPath', return_value=None):
  150. self.assertIsNone(self._event_log_module.Write())
  151. def test_write_non_string(self):
  152. """Test Write() with non-string type for |path| throws TypeError."""
  153. with self.assertRaises(TypeError):
  154. self._event_log_module.Write(path=1234)
  155. if __name__ == '__main__':
  156. unittest.main()