1 | """ |
---|
2 | Utilities for turning objects into human-readable strings. |
---|
3 | |
---|
4 | This module has been ported to Python 3. |
---|
5 | """ |
---|
6 | |
---|
7 | import os |
---|
8 | from reprlib import Repr |
---|
9 | |
---|
10 | class BetterRepr(Repr, object): |
---|
11 | def __init__(self): |
---|
12 | Repr.__init__(self) |
---|
13 | |
---|
14 | # Note: These levels can get adjusted dynamically! My goal is to get more info when printing important debug stuff like exceptions and stack traces and less info when logging normal events. --Zooko 2000-10-14 |
---|
15 | self.maxlevel = 6 |
---|
16 | self.maxdict = 6 |
---|
17 | self.maxlist = 6 |
---|
18 | self.maxtuple = 6 |
---|
19 | self.maxstring = 300 |
---|
20 | self.maxother = 300 |
---|
21 | |
---|
22 | def repr_function(self, obj, level): |
---|
23 | if hasattr(obj, '__code__'): |
---|
24 | return '<' + obj.__name__ + '() at ' + os.path.basename(obj.__code__.co_filename) + ':' + str(obj.__code__.co_firstlineno) + '>' |
---|
25 | else: |
---|
26 | return '<' + obj.__name__ + '() at (builtin)' |
---|
27 | |
---|
28 | def repr_instance_method(self, obj, level): |
---|
29 | if hasattr(obj, '__code__'): |
---|
30 | return '<' + obj.__self__.__class__.__name__ + '.' + obj.__func__.__name__ + '() at ' + os.path.basename(obj.__func__.__code__.co_filename) + ':' + str(obj.__func__.__code__.co_firstlineno) + '>' |
---|
31 | else: |
---|
32 | return '<' + obj.__self__.__class__.__name__ + '.' + obj.__func__.__name__ + '() at (builtin)' |
---|
33 | |
---|
34 | def repr_long(self, obj, level): |
---|
35 | s = repr(obj) # XXX Hope this isn't too slow... |
---|
36 | if len(s) > self.maxlong: |
---|
37 | i = max(0, (self.maxlong-3) // 2) |
---|
38 | j = max(0, self.maxlong-3-i) |
---|
39 | s = s[:i] + '...' + s[len(s)-j:] |
---|
40 | if s[-1] == 'L': |
---|
41 | return s[:-1] |
---|
42 | return s |
---|
43 | |
---|
44 | def repr_instance(self, obj, level): |
---|
45 | """ |
---|
46 | If it is an instance of Exception, format it nicely (trying to emulate |
---|
47 | the format that you see when an exception is actually raised, plus |
---|
48 | bracketing '<''s). If it is an instance of dict call self.repr_dict() |
---|
49 | on it. If it is an instance of list call self.repr_list() on it. Else |
---|
50 | call Repr.repr_instance(). |
---|
51 | """ |
---|
52 | if isinstance(obj, Exception): |
---|
53 | # Don't cut down exception strings so much. |
---|
54 | tms = self.maxstring |
---|
55 | self.maxstring = max(512, tms * 4) |
---|
56 | tml = self.maxlist |
---|
57 | self.maxlist = max(12, tml * 4) |
---|
58 | try: |
---|
59 | if hasattr(obj, 'args'): |
---|
60 | if len(obj.args) == 1: |
---|
61 | return '<' + obj.__class__.__name__ + ': ' + self.repr1(obj.args[0], level-1) + '>' |
---|
62 | else: |
---|
63 | return '<' + obj.__class__.__name__ + ': ' + self.repr1(obj.args, level-1) + '>' |
---|
64 | else: |
---|
65 | return '<' + obj.__class__.__name__ + '>' |
---|
66 | finally: |
---|
67 | self.maxstring = tms |
---|
68 | self.maxlist = tml |
---|
69 | |
---|
70 | if isinstance(obj, dict): |
---|
71 | return self.repr_dict(obj, level) |
---|
72 | |
---|
73 | if isinstance(obj, list): |
---|
74 | return self.repr_list(obj, level) |
---|
75 | |
---|
76 | return Repr.repr_instance(self, obj, level) |
---|
77 | |
---|
78 | def repr_list(self, obj, level): |
---|
79 | """ |
---|
80 | copied from standard repr.py and fixed to work on multithreadedly mutating lists. |
---|
81 | """ |
---|
82 | if level <= 0: return '[...]' |
---|
83 | n = len(obj) |
---|
84 | myl = obj[:min(n, self.maxlist)] |
---|
85 | s = '' |
---|
86 | for item in myl: |
---|
87 | entry = self.repr1(item, level-1) |
---|
88 | if s: s = s + ', ' |
---|
89 | s = s + entry |
---|
90 | if n > self.maxlist: s = s + ', ...' |
---|
91 | return '[' + s + ']' |
---|
92 | |
---|
93 | def repr_dict(self, obj, level): |
---|
94 | """ |
---|
95 | copied from standard repr.py and fixed to work on multithreadedly mutating dicts. |
---|
96 | """ |
---|
97 | if level <= 0: return '{...}' |
---|
98 | s = '' |
---|
99 | n = len(obj) |
---|
100 | items = list(obj.items())[:min(n, self.maxdict)] |
---|
101 | items.sort() |
---|
102 | for key, val in items: |
---|
103 | entry = self.repr1(key, level-1) + ':' + self.repr1(val, level-1) |
---|
104 | if s: s = s + ', ' |
---|
105 | s = s + entry |
---|
106 | if n > self.maxdict: s = s + ', ...' |
---|
107 | return '{' + s + '}' |
---|
108 | |
---|
109 | # This object can be changed by other code updating this module's "brepr" |
---|
110 | # variables. This is so that (a) code can use humanreadable with |
---|
111 | # "from humanreadable import hr; hr(mything)", and (b) code can override |
---|
112 | # humanreadable to provide application-specific human readable output |
---|
113 | # (e.g. libbase32's base32id.AbbrevRepr). |
---|
114 | brepr = BetterRepr() |
---|
115 | |
---|
116 | def hr(x): |
---|
117 | return brepr.repr(x) |
---|