blob: 47c932ac2dae2e9ca0ef8cb91ee03cf2b857463e [file] [log] [blame]
Fred Drakef0193242001-10-12 20:56:29 +00001import _hotshot
2import os.path
3import parser
4import symbol
Fred Drakef0193242001-10-12 20:56:29 +00005
6from _hotshot import \
7 WHAT_ENTER, \
8 WHAT_EXIT, \
9 WHAT_LINENO, \
10 WHAT_DEFINE_FILE, \
Fred Draked5d5a042001-10-15 22:05:32 +000011 WHAT_DEFINE_FUNC, \
Fred Drakef0193242001-10-12 20:56:29 +000012 WHAT_ADD_INFO
13
14
15__all__ = ["LogReader", "ENTER", "EXIT", "LINE"]
16
17
18ENTER = WHAT_ENTER
19EXIT = WHAT_EXIT
20LINE = WHAT_LINENO
21
22
Fred Drakef0193242001-10-12 20:56:29 +000023class LogReader:
24 def __init__(self, logfn):
25 # fileno -> filename
26 self._filemap = {}
27 # (fileno, lineno) -> filename, funcname
28 self._funcmap = {}
29
Fred Draked62f1512001-10-13 02:55:40 +000030 self._reader = _hotshot.logreader(logfn)
31 self._nextitem = self._reader.next
Fred Drake8e26b522001-10-29 20:57:23 +000032 self._info = self._reader.info
Benjamin Peterson6e3dbbd2009-10-09 22:15:50 +000033 if 'current-directory' in self._info:
Fred Drake8b688622002-03-12 14:26:37 +000034 self.cwd = self._info['current-directory']
35 else:
36 self.cwd = None
Fred Drake127ee162002-05-29 19:40:36 +000037
38 # This mirrors the call stack of the profiled code as the log
39 # is read back in. It contains tuples of the form:
40 #
41 # (file name, line number of function def, function name)
42 #
Fred Drakef0193242001-10-12 20:56:29 +000043 self._stack = []
Fred Drake8b688622002-03-12 14:26:37 +000044 self._append = self._stack.append
45 self._pop = self._stack.pop
Fred Drakef0193242001-10-12 20:56:29 +000046
Tim Peters30d48962002-07-18 14:54:28 +000047 def close(self):
48 self._reader.close()
49
Fred Drake7d17e6f2002-07-18 19:17:20 +000050 def fileno(self):
51 """Return the file descriptor of the log reader's log file."""
52 return self._reader.fileno()
53
Fred Drake8e26b522001-10-29 20:57:23 +000054 def addinfo(self, key, value):
55 """This method is called for each additional ADD_INFO record.
56
57 This can be overridden by applications that want to receive
58 these events. The default implementation does not need to be
59 called by alternate implementations.
60
61 The initial set of ADD_INFO records do not pass through this
62 mechanism; this is only needed to receive notification when
63 new values are added. Subclasses can inspect self._info after
64 calling LogReader.__init__().
65 """
66 pass
67
Fred Drake8b688622002-03-12 14:26:37 +000068 def get_filename(self, fileno):
69 try:
70 return self._filemap[fileno]
71 except KeyError:
72 raise ValueError, "unknown fileno"
73
74 def get_filenames(self):
75 return self._filemap.values()
76
77 def get_fileno(self, filename):
78 filename = os.path.normcase(os.path.normpath(filename))
79 for fileno, name in self._filemap.items():
80 if name == filename:
81 return fileno
82 raise ValueError, "unknown filename"
83
84 def get_funcname(self, fileno, lineno):
85 try:
86 return self._funcmap[(fileno, lineno)]
87 except KeyError:
88 raise ValueError, "unknown function location"
89
Fred Drakef0193242001-10-12 20:56:29 +000090 # Iteration support:
91 # This adds an optional (& ignored) parameter to next() so that the
92 # same bound method can be used as the __getitem__() method -- this
93 # avoids using an additional method call which kills the performance.
94
95 def next(self, index=0):
Fred Draked5d5a042001-10-15 22:05:32 +000096 while 1:
Guido van Rossum0d7b8bc2002-07-18 19:48:46 +000097 # This call may raise StopIteration:
Fred Drakefbe36082002-07-18 19:20:23 +000098 what, tdelta, fileno, lineno = self._nextitem()
Fred Drake8b688622002-03-12 14:26:37 +000099
100 # handle the most common cases first
101
102 if what == WHAT_ENTER:
103 filename, funcname = self._decode_location(fileno, lineno)
Fred Drake127ee162002-05-29 19:40:36 +0000104 t = (filename, lineno, funcname)
105 self._append(t)
106 return what, t, tdelta
Fred Drake8b688622002-03-12 14:26:37 +0000107
108 if what == WHAT_EXIT:
Fred Drake127ee162002-05-29 19:40:36 +0000109 return what, self._pop(), tdelta
Fred Drake8b688622002-03-12 14:26:37 +0000110
111 if what == WHAT_LINENO:
Fred Drake127ee162002-05-29 19:40:36 +0000112 filename, firstlineno, funcname = self._stack[-1]
Fred Drake8b688622002-03-12 14:26:37 +0000113 return what, (filename, lineno, funcname), tdelta
114
Fred Draked5d5a042001-10-15 22:05:32 +0000115 if what == WHAT_DEFINE_FILE:
Fred Drake8b688622002-03-12 14:26:37 +0000116 filename = os.path.normcase(os.path.normpath(tdelta))
117 self._filemap[fileno] = filename
118 elif what == WHAT_DEFINE_FUNC:
Fred Draked5d5a042001-10-15 22:05:32 +0000119 filename = self._filemap[fileno]
120 self._funcmap[(fileno, lineno)] = (filename, tdelta)
Fred Drake8b688622002-03-12 14:26:37 +0000121 elif what == WHAT_ADD_INFO:
Fred Drake8e26b522001-10-29 20:57:23 +0000122 # value already loaded into self.info; call the
123 # overridable addinfo() handler so higher-level code
124 # can pick up the new value
Fred Drake8b688622002-03-12 14:26:37 +0000125 if tdelta == 'current-directory':
126 self.cwd = lineno
Fred Drake8e26b522001-10-29 20:57:23 +0000127 self.addinfo(tdelta, lineno)
Fred Draked5d5a042001-10-15 22:05:32 +0000128 else:
Fred Drake8b688622002-03-12 14:26:37 +0000129 raise ValueError, "unknown event type"
Fred Drakef0193242001-10-12 20:56:29 +0000130
Guido van Rossum32616cf2002-07-18 14:33:14 +0000131 def __iter__(self):
132 return self
Fred Drakef0193242001-10-12 20:56:29 +0000133
134 #
135 # helpers
136 #
137
138 def _decode_location(self, fileno, lineno):
139 try:
140 return self._funcmap[(fileno, lineno)]
141 except KeyError:
Fred Draked5d5a042001-10-15 22:05:32 +0000142 #
143 # This should only be needed when the log file does not
144 # contain all the DEFINE_FUNC records needed to allow the
145 # function name to be retrieved from the log file.
146 #
Fred Drakef0193242001-10-12 20:56:29 +0000147 if self._loadfile(fileno):
148 filename = funcname = None
149 try:
150 filename, funcname = self._funcmap[(fileno, lineno)]
151 except KeyError:
152 filename = self._filemap.get(fileno)
153 funcname = None
154 self._funcmap[(fileno, lineno)] = (filename, funcname)
155 return filename, funcname
156
157 def _loadfile(self, fileno):
158 try:
159 filename = self._filemap[fileno]
160 except KeyError:
161 print "Could not identify fileId", fileno
162 return 1
163 if filename is None:
164 return 1
165 absname = os.path.normcase(os.path.join(self.cwd, filename))
166
167 try:
168 fp = open(absname)
169 except IOError:
170 return
171 st = parser.suite(fp.read())
172 fp.close()
173
174 # Scan the tree looking for def and lambda nodes, filling in
175 # self._funcmap with all the available information.
176 funcdef = symbol.funcdef
177 lambdef = symbol.lambdef
178
179 stack = [st.totuple(1)]
180
181 while stack:
182 tree = stack.pop()
183 try:
184 sym = tree[0]
185 except (IndexError, TypeError):
186 continue
187 if sym == funcdef:
188 self._funcmap[(fileno, tree[2][2])] = filename, tree[2][1]
189 elif sym == lambdef:
190 self._funcmap[(fileno, tree[1][2])] = filename, "<lambda>"
191 stack.extend(list(tree[1:]))