| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  | #  Author:      Fred L. Drake, Jr. | 
					
						
							| 
									
										
										
										
											2001-10-09 20:53:48 +00:00
										 |  |  | #               fdrake@acm.org | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | # | 
					
						
							|  |  |  | #  This is a simple little module I wrote to make life easier.  I didn't | 
					
						
							|  |  |  | #  see anything quite like it in the library, though I may have overlooked | 
					
						
							|  |  |  | #  something.  I wrote this when I was trying to read some heavily nested | 
					
						
							| 
									
										
										
										
											2000-07-16 12:04:32 +00:00
										 |  |  | #  tuples with fairly non-descriptive content.  This is modeled very much | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | #  after Lisp/Scheme - style pretty-printing of lists.  If you find it | 
					
						
							|  |  |  | #  useful, thank small children who sleep at night. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """Support to pretty-print lists, tuples, & dictionaries recursively.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Very simple, but useful, especially in debugging data structures. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | Classes | 
					
						
							|  |  |  | ------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PrettyPrinter() | 
					
						
							|  |  |  |     Handle pretty-printing operations onto a stream using a configured | 
					
						
							|  |  |  |     set of formatting parameters. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | Functions | 
					
						
							|  |  |  | --------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pformat() | 
					
						
							|  |  |  |     Format a Python object into a pretty-printed representation. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pprint() | 
					
						
							| 
									
										
										
										
											2004-05-14 16:31:56 +00:00
										 |  |  |     Pretty-print a Python object to a stream [default is sys.stdout]. | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | saferepr() | 
					
						
							|  |  |  |     Generate a 'standard' repr()-like value, but protect against recursive | 
					
						
							|  |  |  |     data structures. | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 08:51:33 +02:00
										 |  |  | import collections as _collections | 
					
						
							| 
									
										
										
										
											2002-12-31 07:14:18 +00:00
										 |  |  | import sys as _sys | 
					
						
							| 
									
										
										
										
											2015-03-24 19:31:50 +02:00
										 |  |  | import types as _types | 
					
						
							| 
									
										
										
										
											2007-08-09 01:03:29 +00:00
										 |  |  | from io import StringIO as _StringIO | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-02-12 02:00:42 +00:00
										 |  |  | __all__ = ["pprint","pformat","isreadable","isrecursive","saferepr", | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  |            "PrettyPrinter", "pp"] | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  | def pprint(object, stream=None, indent=1, width=80, depth=None, *, | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  |            compact=False, sort_dicts=True, underscore_numbers=False): | 
					
						
							| 
									
										
										
										
											2004-05-14 16:31:56 +00:00
										 |  |  |     """Pretty-print a Python object to a stream [default is sys.stdout].""" | 
					
						
							| 
									
										
										
										
											2003-12-03 20:26:05 +00:00
										 |  |  |     printer = PrettyPrinter( | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |         stream=stream, indent=indent, width=width, depth=depth, | 
					
						
							| 
									
										
										
										
											2021-10-21 16:42:55 -04:00
										 |  |  |         compact=compact, sort_dicts=sort_dicts, | 
					
						
							|  |  |  |         underscore_numbers=underscore_numbers) | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  |     printer.pprint(object) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  | def pformat(object, indent=1, width=80, depth=None, *, | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  |             compact=False, sort_dicts=True, underscore_numbers=False): | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  |     """Format a Python object into a pretty-printed representation.""" | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |     return PrettyPrinter(indent=indent, width=width, depth=depth, | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  |                          compact=compact, sort_dicts=sort_dicts, | 
					
						
							|  |  |  |                          underscore_numbers=underscore_numbers).pformat(object) | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  | def pp(object, *args, sort_dicts=False, **kwargs): | 
					
						
							|  |  |  |     """Pretty-print a Python object""" | 
					
						
							|  |  |  |     pprint(object, *args, sort_dicts=sort_dicts, **kwargs) | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-14 07:05:58 +00:00
										 |  |  | def saferepr(object): | 
					
						
							|  |  |  |     """Version of repr() which can handle recursive data structures.""" | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |     return PrettyPrinter()._safe_repr(object, {}, None, 0)[0] | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-07-18 20:42:39 +00:00
										 |  |  | def isreadable(object): | 
					
						
							|  |  |  |     """Determine if saferepr(object) is readable by eval().""" | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |     return PrettyPrinter()._safe_repr(object, {}, None, 0)[1] | 
					
						
							| 
									
										
										
										
											1997-07-18 20:42:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-07-18 20:42:39 +00:00
										 |  |  | def isrecursive(object): | 
					
						
							|  |  |  |     """Determine if object requires a recursive representation.""" | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |     return PrettyPrinter()._safe_repr(object, {}, None, 0)[2] | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-19 01:07:05 +00:00
										 |  |  | class _safe_key: | 
					
						
							|  |  |  |     """Helper function for key functions when sorting unorderable objects.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-17 08:32:47 +03:00
										 |  |  |     The wrapped-object will fallback to a Py2.x style comparison for | 
					
						
							| 
									
										
										
										
											2009-11-19 01:07:05 +00:00
										 |  |  |     unorderable types (sorting first comparing the type name and then by | 
					
						
							|  |  |  |     the obj ids).  Does not work recursively, so dict.items() must have | 
					
						
							|  |  |  |     _safe_key applied to both the key and the value. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     __slots__ = ['obj'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, obj): | 
					
						
							|  |  |  |         self.obj = obj | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __lt__(self, other): | 
					
						
							| 
									
										
										
										
											2012-07-21 11:17:38 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2015-04-06 22:52:44 +03:00
										 |  |  |             return self.obj < other.obj | 
					
						
							| 
									
										
										
										
											2012-07-21 11:17:38 +02:00
										 |  |  |         except TypeError: | 
					
						
							| 
									
										
										
										
											2015-04-06 22:52:44 +03:00
										 |  |  |             return ((str(type(self.obj)), id(self.obj)) < \ | 
					
						
							|  |  |  |                     (str(type(other.obj)), id(other.obj))) | 
					
						
							| 
									
										
										
										
											2009-11-19 01:07:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-19 01:07:05 +00:00
										 |  |  | def _safe_tuple(t): | 
					
						
							|  |  |  |     "Helper function for comparing 2-tuples" | 
					
						
							|  |  |  |     return _safe_key(t[0]), _safe_key(t[1]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | class PrettyPrinter: | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |     def __init__(self, indent=1, width=80, depth=None, stream=None, *, | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  |                  compact=False, sort_dicts=True, underscore_numbers=False): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         """Handle pretty printing operations onto a stream using a set of
 | 
					
						
							|  |  |  |         configured parameters. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         indent | 
					
						
							|  |  |  |             Number of spaces to indent for each level of nesting. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         width | 
					
						
							|  |  |  |             Attempted maximum number of columns in the output. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         depth | 
					
						
							|  |  |  |             The maximum depth to print out nested structures. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         stream | 
					
						
							|  |  |  |             The desired output stream.  If omitted (or false), the standard | 
					
						
							|  |  |  |             output stream available at construction will be used. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |         compact | 
					
						
							|  |  |  |             If true, several items will be combined in one line. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  |         sort_dicts | 
					
						
							|  |  |  |             If true, dict keys are sorted. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-13 15:04:17 +03:00
										 |  |  |         underscore_numbers | 
					
						
							|  |  |  |             If true, digit groups are separated with underscores. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         indent = int(indent) | 
					
						
							|  |  |  |         width = int(width) | 
					
						
							| 
									
										
										
										
											2015-03-26 08:43:21 +02:00
										 |  |  |         if indent < 0: | 
					
						
							|  |  |  |             raise ValueError('indent must be >= 0') | 
					
						
							|  |  |  |         if depth is not None and depth <= 0: | 
					
						
							|  |  |  |             raise ValueError('depth must be > 0') | 
					
						
							|  |  |  |         if not width: | 
					
						
							|  |  |  |             raise ValueError('width must be != 0') | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |         self._depth = depth | 
					
						
							|  |  |  |         self._indent_per_level = indent | 
					
						
							|  |  |  |         self._width = width | 
					
						
							| 
									
										
										
										
											2002-06-01 16:07:16 +00:00
										 |  |  |         if stream is not None: | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |             self._stream = stream | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2002-12-31 07:14:18 +00:00
										 |  |  |             self._stream = _sys.stdout | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |         self._compact = bool(compact) | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  |         self._sort_dicts = sort_dicts | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  |         self._underscore_numbers = underscore_numbers | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def pprint(self, object): | 
					
						
							| 
									
										
										
										
											2021-07-19 10:19:02 +01:00
										 |  |  |         if self._stream is not None: | 
					
						
							|  |  |  |             self._format(object, self._stream, 0, 0, {}, 0) | 
					
						
							|  |  |  |             self._stream.write("\n") | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def pformat(self, object): | 
					
						
							| 
									
										
										
										
											2002-12-31 07:14:18 +00:00
										 |  |  |         sio = _StringIO() | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |         self._format(object, sio, 0, 0, {}, 0) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         return sio.getvalue() | 
					
						
							| 
									
										
										
										
											1997-04-16 16:59:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-07-18 20:42:39 +00:00
										 |  |  |     def isrecursive(self, object): | 
					
						
							| 
									
										
										
										
											2002-12-31 07:14:18 +00:00
										 |  |  |         return self.format(object, {}, 0, 0)[2] | 
					
						
							| 
									
										
										
										
											1997-07-18 20:42:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def isreadable(self, object): | 
					
						
							| 
									
										
										
										
											2002-12-31 07:14:18 +00:00
										 |  |  |         s, readable, recursive = self.format(object, {}, 0, 0) | 
					
						
							| 
									
										
										
										
											2002-04-02 05:08:35 +00:00
										 |  |  |         return readable and not recursive | 
					
						
							| 
									
										
										
										
											1997-07-18 20:42:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |     def _format(self, object, stream, indent, allowance, context, level): | 
					
						
							| 
									
										
										
										
											2013-10-03 21:29:36 +02:00
										 |  |  |         objid = id(object) | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  |         if objid in context: | 
					
						
							|  |  |  |             stream.write(_recursion(object)) | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |             self._recursive = True | 
					
						
							|  |  |  |             self._readable = False | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  |             return | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |         rep = self._repr(object, context, level) | 
					
						
							| 
									
										
										
										
											2015-02-14 10:55:19 +02:00
										 |  |  |         max_width = self._width - indent - allowance | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |         if len(rep) > max_width: | 
					
						
							|  |  |  |             p = self._dispatch.get(type(object).__repr__, None) | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  |             # Lazy import to improve module import time | 
					
						
							|  |  |  |             from dataclasses import is_dataclass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |             if p is not None: | 
					
						
							|  |  |  |                 context[objid] = 1 | 
					
						
							|  |  |  |                 p(self, object, stream, indent, allowance, context, level + 1) | 
					
						
							|  |  |  |                 del context[objid] | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  |                 return | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  |             elif (is_dataclass(object) and | 
					
						
							| 
									
										
										
										
											2021-04-14 00:59:24 +01:00
										 |  |  |                   not isinstance(object, type) and | 
					
						
							|  |  |  |                   object.__dataclass_params__.repr and | 
					
						
							|  |  |  |                   # Check dataclass has generated repr method. | 
					
						
							|  |  |  |                   hasattr(object.__repr__, "__wrapped__") and | 
					
						
							|  |  |  |                   "__create_fn__" in object.__repr__.__wrapped__.__qualname__): | 
					
						
							|  |  |  |                 context[objid] = 1 | 
					
						
							|  |  |  |                 self._pprint_dataclass(object, stream, indent, allowance, context, level + 1) | 
					
						
							|  |  |  |                 del context[objid] | 
					
						
							|  |  |  |                 return | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |         stream.write(rep) | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 00:59:24 +01:00
										 |  |  |     def _pprint_dataclass(self, object, stream, indent, allowance, context, level): | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  |         # Lazy import to improve module import time | 
					
						
							|  |  |  |         from dataclasses import fields as dataclass_fields | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 00:59:24 +01:00
										 |  |  |         cls_name = object.__class__.__name__ | 
					
						
							|  |  |  |         indent += len(cls_name) + 1 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  |         items = [(f.name, getattr(object, f.name)) for f in dataclass_fields(object) if f.repr] | 
					
						
							| 
									
										
										
										
											2021-04-14 00:59:24 +01:00
										 |  |  |         stream.write(cls_name + '(') | 
					
						
							|  |  |  |         self._format_namespace_items(items, stream, indent, allowance, context, level) | 
					
						
							|  |  |  |         stream.write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |     _dispatch = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_dict(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         write('{') | 
					
						
							|  |  |  |         if self._indent_per_level > 1: | 
					
						
							|  |  |  |             write((self._indent_per_level - 1) * ' ') | 
					
						
							|  |  |  |         length = len(object) | 
					
						
							|  |  |  |         if length: | 
					
						
							| 
									
										
										
										
											2019-03-22 18:22:20 +01:00
										 |  |  |             if self._sort_dicts: | 
					
						
							|  |  |  |                 items = sorted(object.items(), key=_safe_tuple) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 items = object.items() | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |             self._format_dict_items(items, stream, indent, allowance + 1, | 
					
						
							|  |  |  |                                     context, level) | 
					
						
							|  |  |  |         write('}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[dict.__repr__] = _pprint_dict | 
					
						
							| 
									
										
										
										
											2015-03-26 08:51:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_ordered_dict(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if not len(object): | 
					
						
							|  |  |  |             stream.write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         cls = object.__class__ | 
					
						
							|  |  |  |         stream.write(cls.__name__ + '(') | 
					
						
							|  |  |  |         self._format(list(object.items()), stream, | 
					
						
							|  |  |  |                      indent + len(cls.__name__) + 1, allowance + 1, | 
					
						
							|  |  |  |                      context, level) | 
					
						
							|  |  |  |         stream.write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.OrderedDict.__repr__] = _pprint_ordered_dict | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-20 16:30:00 -03:00
										 |  |  |     def _pprint_dict_view(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         """Pretty print dict views (keys, values, items).""" | 
					
						
							|  |  |  |         if isinstance(object, self._dict_items_view): | 
					
						
							|  |  |  |             key = _safe_tuple | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             key = _safe_key | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         write(object.__class__.__name__ + '([') | 
					
						
							|  |  |  |         if self._indent_per_level > 1: | 
					
						
							|  |  |  |             write((self._indent_per_level - 1) * ' ') | 
					
						
							|  |  |  |         length = len(object) | 
					
						
							|  |  |  |         if length: | 
					
						
							|  |  |  |             if self._sort_dicts: | 
					
						
							|  |  |  |                 entries = sorted(object, key=key) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 entries = object | 
					
						
							|  |  |  |             self._format_items(entries, stream, indent, allowance + 1, | 
					
						
							|  |  |  |                                context, level) | 
					
						
							|  |  |  |         write('])') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_mapping_abc_view(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         """Pretty print mapping views from collections.abc.""" | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         write(object.__class__.__name__ + '(') | 
					
						
							|  |  |  |         # Dispatch formatting to the view's _mapping | 
					
						
							|  |  |  |         self._format(object._mapping, stream, indent, allowance, context, level) | 
					
						
							|  |  |  |         write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dict_keys_view = type({}.keys()) | 
					
						
							|  |  |  |     _dispatch[_dict_keys_view.__repr__] = _pprint_dict_view | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dict_values_view = type({}.values()) | 
					
						
							|  |  |  |     _dispatch[_dict_values_view.__repr__] = _pprint_dict_view | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dict_items_view = type({}.items()) | 
					
						
							|  |  |  |     _dispatch[_dict_items_view.__repr__] = _pprint_dict_view | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.abc.MappingView.__repr__] = _pprint_mapping_abc_view | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _view_reprs = {cls.__repr__ for cls in | 
					
						
							|  |  |  |                    (_dict_keys_view, _dict_values_view, _dict_items_view, | 
					
						
							|  |  |  |                     _collections.abc.MappingView)} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |     def _pprint_list(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         stream.write('[') | 
					
						
							|  |  |  |         self._format_items(object, stream, indent, allowance + 1, | 
					
						
							|  |  |  |                            context, level) | 
					
						
							|  |  |  |         stream.write(']') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[list.__repr__] = _pprint_list | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_tuple(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         stream.write('(') | 
					
						
							|  |  |  |         endchar = ',)' if len(object) == 1 else ')' | 
					
						
							|  |  |  |         self._format_items(object, stream, indent, allowance + len(endchar), | 
					
						
							|  |  |  |                            context, level) | 
					
						
							|  |  |  |         stream.write(endchar) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[tuple.__repr__] = _pprint_tuple | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_set(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if not len(object): | 
					
						
							|  |  |  |             stream.write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         typ = object.__class__ | 
					
						
							|  |  |  |         if typ is set: | 
					
						
							|  |  |  |             stream.write('{') | 
					
						
							|  |  |  |             endchar = '}' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             stream.write(typ.__name__ + '({') | 
					
						
							|  |  |  |             endchar = '})' | 
					
						
							|  |  |  |             indent += len(typ.__name__) + 1 | 
					
						
							|  |  |  |         object = sorted(object, key=_safe_key) | 
					
						
							|  |  |  |         self._format_items(object, stream, indent, allowance + len(endchar), | 
					
						
							|  |  |  |                            context, level) | 
					
						
							|  |  |  |         stream.write(endchar) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[set.__repr__] = _pprint_set | 
					
						
							|  |  |  |     _dispatch[frozenset.__repr__] = _pprint_set | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_str(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         if not len(object): | 
					
						
							|  |  |  |             write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         chunks = [] | 
					
						
							|  |  |  |         lines = object.splitlines(True) | 
					
						
							|  |  |  |         if level == 1: | 
					
						
							|  |  |  |             indent += 1 | 
					
						
							|  |  |  |             allowance += 1 | 
					
						
							|  |  |  |         max_width1 = max_width = self._width - indent | 
					
						
							|  |  |  |         for i, line in enumerate(lines): | 
					
						
							|  |  |  |             rep = repr(line) | 
					
						
							|  |  |  |             if i == len(lines) - 1: | 
					
						
							|  |  |  |                 max_width1 -= allowance | 
					
						
							|  |  |  |             if len(rep) <= max_width1: | 
					
						
							|  |  |  |                 chunks.append(rep) | 
					
						
							|  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  |                 # Lazy import to improve module import time | 
					
						
							|  |  |  |                 import re | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |                 # A list of alternating (non-space, space) strings | 
					
						
							|  |  |  |                 parts = re.findall(r'\S*\s*', line) | 
					
						
							|  |  |  |                 assert parts | 
					
						
							|  |  |  |                 assert not parts[-1] | 
					
						
							|  |  |  |                 parts.pop()  # drop empty last part | 
					
						
							|  |  |  |                 max_width2 = max_width | 
					
						
							|  |  |  |                 current = '' | 
					
						
							|  |  |  |                 for j, part in enumerate(parts): | 
					
						
							|  |  |  |                     candidate = current + part | 
					
						
							|  |  |  |                     if j == len(parts) - 1 and i == len(lines) - 1: | 
					
						
							|  |  |  |                         max_width2 -= allowance | 
					
						
							|  |  |  |                     if len(repr(candidate)) > max_width2: | 
					
						
							| 
									
										
										
										
											2014-12-20 20:57:15 +02:00
										 |  |  |                         if current: | 
					
						
							|  |  |  |                             chunks.append(repr(current)) | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |                         current = part | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         current = candidate | 
					
						
							|  |  |  |                 if current: | 
					
						
							|  |  |  |                     chunks.append(repr(current)) | 
					
						
							|  |  |  |         if len(chunks) == 1: | 
					
						
							|  |  |  |             write(rep) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         if level == 1: | 
					
						
							|  |  |  |             write('(') | 
					
						
							|  |  |  |         for i, rep in enumerate(chunks): | 
					
						
							|  |  |  |             if i > 0: | 
					
						
							|  |  |  |                 write('\n' + ' '*indent) | 
					
						
							|  |  |  |             write(rep) | 
					
						
							|  |  |  |         if level == 1: | 
					
						
							|  |  |  |             write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[str.__repr__] = _pprint_str | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 19:22:37 +02:00
										 |  |  |     def _pprint_bytes(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         if len(object) <= 4: | 
					
						
							|  |  |  |             write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         parens = level == 1 | 
					
						
							|  |  |  |         if parens: | 
					
						
							|  |  |  |             indent += 1 | 
					
						
							|  |  |  |             allowance += 1 | 
					
						
							|  |  |  |             write('(') | 
					
						
							|  |  |  |         delim = '' | 
					
						
							|  |  |  |         for rep in _wrap_bytes_repr(object, self._width - indent, allowance): | 
					
						
							|  |  |  |             write(delim) | 
					
						
							|  |  |  |             write(rep) | 
					
						
							|  |  |  |             if not delim: | 
					
						
							|  |  |  |                 delim = '\n' + ' '*indent | 
					
						
							|  |  |  |         if parens: | 
					
						
							|  |  |  |             write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[bytes.__repr__] = _pprint_bytes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_bytearray(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         write('bytearray(') | 
					
						
							|  |  |  |         self._pprint_bytes(bytes(object), stream, indent + 10, | 
					
						
							|  |  |  |                            allowance + 1, context, level + 1) | 
					
						
							|  |  |  |         write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[bytearray.__repr__] = _pprint_bytearray | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 19:31:50 +02:00
										 |  |  |     def _pprint_mappingproxy(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         stream.write('mappingproxy(') | 
					
						
							|  |  |  |         self._format(object.copy(), stream, indent + 13, allowance + 1, | 
					
						
							|  |  |  |                      context, level) | 
					
						
							|  |  |  |         stream.write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_types.MappingProxyType.__repr__] = _pprint_mappingproxy | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-27 01:13:18 +02:00
										 |  |  |     def _pprint_simplenamespace(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if type(object) is _types.SimpleNamespace: | 
					
						
							|  |  |  |             # The SimpleNamespace repr is "namespace" instead of the class | 
					
						
							|  |  |  |             # name, so we do the same here. For subclasses; use the class name. | 
					
						
							|  |  |  |             cls_name = 'namespace' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cls_name = object.__class__.__name__ | 
					
						
							|  |  |  |         indent += len(cls_name) + 1 | 
					
						
							|  |  |  |         items = object.__dict__.items() | 
					
						
							|  |  |  |         stream.write(cls_name + '(') | 
					
						
							| 
									
										
										
										
											2021-04-14 00:59:24 +01:00
										 |  |  |         self._format_namespace_items(items, stream, indent, allowance, context, level) | 
					
						
							| 
									
										
										
										
											2019-06-27 01:13:18 +02:00
										 |  |  |         stream.write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_types.SimpleNamespace.__repr__] = _pprint_simplenamespace | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-14 10:55:19 +02:00
										 |  |  |     def _format_dict_items(self, items, stream, indent, allowance, context, | 
					
						
							|  |  |  |                            level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |         indent += self._indent_per_level | 
					
						
							| 
									
										
										
										
											2015-02-14 10:55:19 +02:00
										 |  |  |         delimnl = ',\n' + ' ' * indent | 
					
						
							|  |  |  |         last_index = len(items) - 1 | 
					
						
							|  |  |  |         for i, (key, ent) in enumerate(items): | 
					
						
							|  |  |  |             last = i == last_index | 
					
						
							|  |  |  |             rep = self._repr(key, context, level) | 
					
						
							|  |  |  |             write(rep) | 
					
						
							|  |  |  |             write(': ') | 
					
						
							|  |  |  |             self._format(ent, stream, indent + len(rep) + 2, | 
					
						
							|  |  |  |                          allowance if last else 1, | 
					
						
							|  |  |  |                          context, level) | 
					
						
							|  |  |  |             if not last: | 
					
						
							|  |  |  |                 write(delimnl) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 00:59:24 +01:00
										 |  |  |     def _format_namespace_items(self, items, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							|  |  |  |         delimnl = ',\n' + ' ' * indent | 
					
						
							|  |  |  |         last_index = len(items) - 1 | 
					
						
							|  |  |  |         for i, (key, ent) in enumerate(items): | 
					
						
							|  |  |  |             last = i == last_index | 
					
						
							|  |  |  |             write(key) | 
					
						
							|  |  |  |             write('=') | 
					
						
							|  |  |  |             if id(ent) in context: | 
					
						
							|  |  |  |                 # Special-case representation of recursion to match standard | 
					
						
							|  |  |  |                 # recursive dataclass repr. | 
					
						
							|  |  |  |                 write("...") | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self._format(ent, stream, indent + len(key) + 1, | 
					
						
							|  |  |  |                              allowance if last else 1, | 
					
						
							|  |  |  |                              context, level) | 
					
						
							|  |  |  |             if not last: | 
					
						
							|  |  |  |                 write(delimnl) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |     def _format_items(self, items, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         write = stream.write | 
					
						
							| 
									
										
										
										
											2015-03-24 18:45:23 +02:00
										 |  |  |         indent += self._indent_per_level | 
					
						
							|  |  |  |         if self._indent_per_level > 1: | 
					
						
							|  |  |  |             write((self._indent_per_level - 1) * ' ') | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |         delimnl = ',\n' + ' ' * indent | 
					
						
							|  |  |  |         delim = '' | 
					
						
							| 
									
										
										
										
											2015-02-14 10:55:19 +02:00
										 |  |  |         width = max_width = self._width - indent + 1 | 
					
						
							|  |  |  |         it = iter(items) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             next_ent = next(it) | 
					
						
							|  |  |  |         except StopIteration: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         last = False | 
					
						
							|  |  |  |         while not last: | 
					
						
							|  |  |  |             ent = next_ent | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 next_ent = next(it) | 
					
						
							|  |  |  |             except StopIteration: | 
					
						
							|  |  |  |                 last = True | 
					
						
							|  |  |  |                 max_width -= allowance | 
					
						
							|  |  |  |                 width -= allowance | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |             if self._compact: | 
					
						
							|  |  |  |                 rep = self._repr(ent, context, level) | 
					
						
							| 
									
										
										
										
											2013-10-03 21:29:36 +02:00
										 |  |  |                 w = len(rep) + 2 | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  |                 if width < w: | 
					
						
							|  |  |  |                     width = max_width | 
					
						
							|  |  |  |                     if delim: | 
					
						
							|  |  |  |                         delim = delimnl | 
					
						
							|  |  |  |                 if width >= w: | 
					
						
							|  |  |  |                     width -= w | 
					
						
							|  |  |  |                     write(delim) | 
					
						
							|  |  |  |                     delim = ', ' | 
					
						
							|  |  |  |                     write(rep) | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |             write(delim) | 
					
						
							|  |  |  |             delim = delimnl | 
					
						
							| 
									
										
										
										
											2015-02-14 10:55:19 +02:00
										 |  |  |             self._format(ent, stream, indent, | 
					
						
							|  |  |  |                          allowance if last else 1, | 
					
						
							|  |  |  |                          context, level) | 
					
						
							| 
									
										
										
										
											2013-10-02 11:56:18 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |     def _repr(self, object, context, level): | 
					
						
							| 
									
										
										
										
											2002-04-02 05:08:35 +00:00
										 |  |  |         repr, readable, recursive = self.format(object, context.copy(), | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |                                                 self._depth, level) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         if not readable: | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |             self._readable = False | 
					
						
							| 
									
										
										
										
											2001-05-14 07:05:58 +00:00
										 |  |  |         if recursive: | 
					
						
							| 
									
										
										
										
											2002-07-08 12:28:06 +00:00
										 |  |  |             self._recursive = True | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         return repr | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-04-02 05:08:35 +00:00
										 |  |  |     def format(self, object, context, maxlevels, level): | 
					
						
							|  |  |  |         """Format object for a specific context, returning a string
 | 
					
						
							|  |  |  |         and flags indicating whether the representation is 'readable' | 
					
						
							|  |  |  |         and whether the object represents a recursive construct. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |         return self._safe_repr(object, context, maxlevels, level) | 
					
						
							| 
									
										
										
										
											2002-04-02 05:08:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 13:35:48 +03:00
										 |  |  |     def _pprint_default_dict(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if not len(object): | 
					
						
							|  |  |  |             stream.write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         rdf = self._repr(object.default_factory, context, level) | 
					
						
							|  |  |  |         cls = object.__class__ | 
					
						
							|  |  |  |         indent += len(cls.__name__) + 1 | 
					
						
							|  |  |  |         stream.write('%s(%s,\n%s' % (cls.__name__, rdf, ' ' * indent)) | 
					
						
							|  |  |  |         self._pprint_dict(object, stream, indent, allowance + 1, context, level) | 
					
						
							|  |  |  |         stream.write(')') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.defaultdict.__repr__] = _pprint_default_dict | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_counter(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if not len(object): | 
					
						
							|  |  |  |             stream.write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         cls = object.__class__ | 
					
						
							|  |  |  |         stream.write(cls.__name__ + '({') | 
					
						
							|  |  |  |         if self._indent_per_level > 1: | 
					
						
							|  |  |  |             stream.write((self._indent_per_level - 1) * ' ') | 
					
						
							|  |  |  |         items = object.most_common() | 
					
						
							|  |  |  |         self._format_dict_items(items, stream, | 
					
						
							|  |  |  |                                 indent + len(cls.__name__) + 1, allowance + 2, | 
					
						
							|  |  |  |                                 context, level) | 
					
						
							|  |  |  |         stream.write('})') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.Counter.__repr__] = _pprint_counter | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_chain_map(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if not len(object.maps): | 
					
						
							|  |  |  |             stream.write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         cls = object.__class__ | 
					
						
							|  |  |  |         stream.write(cls.__name__ + '(') | 
					
						
							|  |  |  |         indent += len(cls.__name__) + 1 | 
					
						
							|  |  |  |         for i, m in enumerate(object.maps): | 
					
						
							|  |  |  |             if i == len(object.maps) - 1: | 
					
						
							|  |  |  |                 self._format(m, stream, indent, allowance + 1, context, level) | 
					
						
							|  |  |  |                 stream.write(')') | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self._format(m, stream, indent, 1, context, level) | 
					
						
							|  |  |  |                 stream.write(',\n' + ' ' * indent) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.ChainMap.__repr__] = _pprint_chain_map | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_deque(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         if not len(object): | 
					
						
							|  |  |  |             stream.write(repr(object)) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         cls = object.__class__ | 
					
						
							|  |  |  |         stream.write(cls.__name__ + '(') | 
					
						
							|  |  |  |         indent += len(cls.__name__) + 1 | 
					
						
							|  |  |  |         stream.write('[') | 
					
						
							|  |  |  |         if object.maxlen is None: | 
					
						
							|  |  |  |             self._format_items(object, stream, indent, allowance + 2, | 
					
						
							|  |  |  |                                context, level) | 
					
						
							|  |  |  |             stream.write('])') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self._format_items(object, stream, indent, 2, | 
					
						
							|  |  |  |                                context, level) | 
					
						
							|  |  |  |             rml = self._repr(object.maxlen, context, level) | 
					
						
							|  |  |  |             stream.write('],\n%smaxlen=%s)' % (' ' * indent, rml)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.deque.__repr__] = _pprint_deque | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_user_dict(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         self._format(object.data, stream, indent, allowance, context, level - 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.UserDict.__repr__] = _pprint_user_dict | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_user_list(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         self._format(object.data, stream, indent, allowance, context, level - 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.UserList.__repr__] = _pprint_user_list | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _pprint_user_string(self, object, stream, indent, allowance, context, level): | 
					
						
							|  |  |  |         self._format(object.data, stream, indent, allowance, context, level - 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _dispatch[_collections.UserString.__repr__] = _pprint_user_string | 
					
						
							| 
									
										
										
										
											2002-04-02 05:08:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |     def _safe_repr(self, object, context, maxlevels, level): | 
					
						
							|  |  |  |         # Return triple (repr_string, isreadable, isrecursive). | 
					
						
							|  |  |  |         typ = type(object) | 
					
						
							|  |  |  |         if typ in _builtin_scalars: | 
					
						
							|  |  |  |             return repr(object), True, False | 
					
						
							| 
									
										
										
										
											1997-04-16 00:49:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |         r = getattr(typ, "__repr__", None) | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if issubclass(typ, int) and r is int.__repr__: | 
					
						
							|  |  |  |             if self._underscore_numbers: | 
					
						
							|  |  |  |                 return f"{object:_d}", True, False | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return repr(object), True, False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |         if issubclass(typ, dict) and r is dict.__repr__: | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  |             if not object: | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |                 return "{}", True, False | 
					
						
							|  |  |  |             objid = id(object) | 
					
						
							|  |  |  |             if maxlevels and level >= maxlevels: | 
					
						
							|  |  |  |                 return "{...}", False, objid in context | 
					
						
							|  |  |  |             if objid in context: | 
					
						
							|  |  |  |                 return _recursion(object), False, True | 
					
						
							|  |  |  |             context[objid] = 1 | 
					
						
							|  |  |  |             readable = True | 
					
						
							|  |  |  |             recursive = False | 
					
						
							|  |  |  |             components = [] | 
					
						
							|  |  |  |             append = components.append | 
					
						
							|  |  |  |             level += 1 | 
					
						
							|  |  |  |             if self._sort_dicts: | 
					
						
							|  |  |  |                 items = sorted(object.items(), key=_safe_tuple) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 items = object.items() | 
					
						
							|  |  |  |             for k, v in items: | 
					
						
							|  |  |  |                 krepr, kreadable, krecur = self.format( | 
					
						
							|  |  |  |                     k, context, maxlevels, level) | 
					
						
							|  |  |  |                 vrepr, vreadable, vrecur = self.format( | 
					
						
							|  |  |  |                     v, context, maxlevels, level) | 
					
						
							|  |  |  |                 append("%s: %s" % (krepr, vrepr)) | 
					
						
							|  |  |  |                 readable = readable and kreadable and vreadable | 
					
						
							|  |  |  |                 if krecur or vrecur: | 
					
						
							|  |  |  |                     recursive = True | 
					
						
							|  |  |  |             del context[objid] | 
					
						
							|  |  |  |             return "{%s}" % ", ".join(components), readable, recursive | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 14:41:41 -07:00
										 |  |  |         if (issubclass(typ, list) and r is list.__repr__) or \ | 
					
						
							|  |  |  |            (issubclass(typ, tuple) and r is tuple.__repr__): | 
					
						
							|  |  |  |             if issubclass(typ, list): | 
					
						
							|  |  |  |                 if not object: | 
					
						
							|  |  |  |                     return "[]", True, False | 
					
						
							|  |  |  |                 format = "[%s]" | 
					
						
							|  |  |  |             elif len(object) == 1: | 
					
						
							|  |  |  |                 format = "(%s,)" | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 if not object: | 
					
						
							|  |  |  |                     return "()", True, False | 
					
						
							|  |  |  |                 format = "(%s)" | 
					
						
							|  |  |  |             objid = id(object) | 
					
						
							|  |  |  |             if maxlevels and level >= maxlevels: | 
					
						
							|  |  |  |                 return format % "...", False, objid in context | 
					
						
							|  |  |  |             if objid in context: | 
					
						
							|  |  |  |                 return _recursion(object), False, True | 
					
						
							|  |  |  |             context[objid] = 1 | 
					
						
							|  |  |  |             readable = True | 
					
						
							|  |  |  |             recursive = False | 
					
						
							|  |  |  |             components = [] | 
					
						
							|  |  |  |             append = components.append | 
					
						
							|  |  |  |             level += 1 | 
					
						
							|  |  |  |             for o in object: | 
					
						
							|  |  |  |                 orepr, oreadable, orecur = self.format( | 
					
						
							|  |  |  |                     o, context, maxlevels, level) | 
					
						
							|  |  |  |                 append(orepr) | 
					
						
							|  |  |  |                 if not oreadable: | 
					
						
							|  |  |  |                     readable = False | 
					
						
							|  |  |  |                 if orecur: | 
					
						
							|  |  |  |                     recursive = True | 
					
						
							|  |  |  |             del context[objid] | 
					
						
							|  |  |  |             return format % ", ".join(components), readable, recursive | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-20 16:30:00 -03:00
										 |  |  |         if issubclass(typ, _collections.abc.MappingView) and r in self._view_reprs: | 
					
						
							|  |  |  |             objid = id(object) | 
					
						
							|  |  |  |             if maxlevels and level >= maxlevels: | 
					
						
							|  |  |  |                 return "{...}", False, objid in context | 
					
						
							|  |  |  |             if objid in context: | 
					
						
							|  |  |  |                 return _recursion(object), False, True | 
					
						
							|  |  |  |             key = _safe_key | 
					
						
							|  |  |  |             if issubclass(typ, (self._dict_items_view, _collections.abc.ItemsView)): | 
					
						
							|  |  |  |                 key = _safe_tuple | 
					
						
							|  |  |  |             if hasattr(object, "_mapping"): | 
					
						
							|  |  |  |                 # Dispatch formatting to the view's _mapping | 
					
						
							|  |  |  |                 mapping_repr, readable, recursive = self.format( | 
					
						
							|  |  |  |                     object._mapping, context, maxlevels, level) | 
					
						
							|  |  |  |                 return (typ.__name__ + '(%s)' % mapping_repr), readable, recursive | 
					
						
							|  |  |  |             elif hasattr(typ, "_mapping"): | 
					
						
							|  |  |  |                 #  We have a view that somehow has lost its type's _mapping, raise | 
					
						
							|  |  |  |                 #  an error by calling repr() instead of failing cryptically later | 
					
						
							|  |  |  |                 return repr(object), True, False | 
					
						
							|  |  |  |             if self._sort_dicts: | 
					
						
							|  |  |  |                 object = sorted(object, key=key) | 
					
						
							|  |  |  |             context[objid] = 1 | 
					
						
							|  |  |  |             readable = True | 
					
						
							|  |  |  |             recursive = False | 
					
						
							|  |  |  |             components = [] | 
					
						
							|  |  |  |             append = components.append | 
					
						
							|  |  |  |             level += 1 | 
					
						
							|  |  |  |             for val in object: | 
					
						
							|  |  |  |                 vrepr, vreadable, vrecur = self.format( | 
					
						
							|  |  |  |                     val, context, maxlevels, level) | 
					
						
							|  |  |  |                 append(vrepr) | 
					
						
							|  |  |  |                 readable = readable and vreadable | 
					
						
							|  |  |  |                 if vrecur: | 
					
						
							|  |  |  |                     recursive = True | 
					
						
							|  |  |  |             del context[objid] | 
					
						
							|  |  |  |             return typ.__name__ + '([%s])' % ", ".join(components), readable, recursive | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 13:31:31 +00:00
										 |  |  |         rep = repr(object) | 
					
						
							|  |  |  |         return rep, (rep and not rep.startswith('<')), False | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-24 09:23:20 +01:00
										 |  |  | _builtin_scalars = frozenset({str, bytes, bytearray, float, complex, | 
					
						
							| 
									
										
										
										
											2015-05-16 21:38:05 +03:00
										 |  |  |                               bool, type(None)}) | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 22:46:54 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  | def _recursion(object): | 
					
						
							|  |  |  |     return ("<Recursion on %s with id=%s>" | 
					
						
							| 
									
										
										
										
											2013-10-03 21:29:36 +02:00
										 |  |  |             % (type(object).__name__, id(object))) | 
					
						
							| 
									
										
										
										
											2001-11-01 17:50:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 19:22:37 +02:00
										 |  |  | def _wrap_bytes_repr(object, width, allowance): | 
					
						
							|  |  |  |     current = b'' | 
					
						
							|  |  |  |     last = len(object) // 4 * 4 | 
					
						
							|  |  |  |     for i in range(0, len(object), 4): | 
					
						
							|  |  |  |         part = object[i: i+4] | 
					
						
							|  |  |  |         candidate = current + part | 
					
						
							|  |  |  |         if i == last: | 
					
						
							|  |  |  |             width -= allowance | 
					
						
							|  |  |  |         if len(repr(candidate)) > width: | 
					
						
							|  |  |  |             if current: | 
					
						
							|  |  |  |                 yield repr(current) | 
					
						
							|  |  |  |             current = part | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             current = candidate | 
					
						
							|  |  |  |     if current: | 
					
						
							|  |  |  |         yield repr(current) |