truncate.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. """
  2. Utilities for truncating assertion output.
  3. Current default behaviour is to truncate assertion explanations at
  4. ~8 terminal lines, unless running in "-vv" mode or running on CI.
  5. """
  6. from __future__ import absolute_import, division, print_function
  7. import os
  8. import six
  9. DEFAULT_MAX_LINES = 8
  10. DEFAULT_MAX_CHARS = 8 * 80
  11. USAGE_MSG = "use '-vv' to show"
  12. def truncate_if_required(explanation, item, max_length=None):
  13. """
  14. Truncate this assertion explanation if the given test item is eligible.
  15. """
  16. if _should_truncate_item(item):
  17. return _truncate_explanation(explanation)
  18. return explanation
  19. def _should_truncate_item(item):
  20. """
  21. Whether or not this test item is eligible for truncation.
  22. """
  23. verbose = item.config.option.verbose
  24. return verbose < 2 and not _running_on_ci()
  25. def _running_on_ci():
  26. """Check if we're currently running on a CI system."""
  27. env_vars = ["CI", "BUILD_NUMBER"]
  28. return any(var in os.environ for var in env_vars)
  29. def _truncate_explanation(input_lines, max_lines=None, max_chars=None):
  30. """
  31. Truncate given list of strings that makes up the assertion explanation.
  32. Truncates to either 8 lines, or 640 characters - whichever the input reaches
  33. first. The remaining lines will be replaced by a usage message.
  34. """
  35. if max_lines is None:
  36. max_lines = DEFAULT_MAX_LINES
  37. if max_chars is None:
  38. max_chars = DEFAULT_MAX_CHARS
  39. # Check if truncation required
  40. input_char_count = len("".join(input_lines))
  41. if len(input_lines) <= max_lines and input_char_count <= max_chars:
  42. return input_lines
  43. # Truncate first to max_lines, and then truncate to max_chars if max_chars
  44. # is exceeded.
  45. truncated_explanation = input_lines[:max_lines]
  46. truncated_explanation = _truncate_by_char_count(truncated_explanation, max_chars)
  47. # Add ellipsis to final line
  48. truncated_explanation[-1] = truncated_explanation[-1] + "..."
  49. # Append useful message to explanation
  50. truncated_line_count = len(input_lines) - len(truncated_explanation)
  51. truncated_line_count += 1 # Account for the part-truncated final line
  52. msg = "...Full output truncated"
  53. if truncated_line_count == 1:
  54. msg += " ({} line hidden)".format(truncated_line_count)
  55. else:
  56. msg += " ({} lines hidden)".format(truncated_line_count)
  57. msg += ", {}".format(USAGE_MSG)
  58. truncated_explanation.extend([six.text_type(""), six.text_type(msg)])
  59. return truncated_explanation
  60. def _truncate_by_char_count(input_lines, max_chars):
  61. # Check if truncation required
  62. if len("".join(input_lines)) <= max_chars:
  63. return input_lines
  64. # Find point at which input length exceeds total allowed length
  65. iterated_char_count = 0
  66. for iterated_index, input_line in enumerate(input_lines):
  67. if iterated_char_count + len(input_line) > max_chars:
  68. break
  69. iterated_char_count += len(input_line)
  70. # Create truncated explanation with modified final line
  71. truncated_result = input_lines[:iterated_index]
  72. final_line = input_lines[iterated_index]
  73. if final_line:
  74. final_line_truncate_point = max_chars - iterated_char_count
  75. final_line = final_line[:final_line_truncate_point]
  76. truncated_result.append(final_line)
  77. return truncated_result