mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 07:31:38 +00:00 
			
		
		
		
	
		
			
	
	
		
			221 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			221 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| 
								 | 
							
								# Print tracebacks, with a dump of local variables.
							 | 
						||
| 
								 | 
							
								# Also an interactive stack trace browser.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import sys
							 | 
						||
| 
								 | 
							
								try:
							 | 
						||
| 
								 | 
							
									import mac
							 | 
						||
| 
								 | 
							
									os = mac
							 | 
						||
| 
								 | 
							
								except NameError:
							 | 
						||
| 
								 | 
							
									import posix
							 | 
						||
| 
								 | 
							
									os = posix
							 | 
						||
| 
								 | 
							
								from stat import *
							 | 
						||
| 
								 | 
							
								import string
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def br(): browser(sys.last_traceback)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def tb(): printtb(sys.last_traceback)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def browser(tb):
							 | 
						||
| 
								 | 
							
									if not tb:
							 | 
						||
| 
								 | 
							
										print 'No traceback.'
							 | 
						||
| 
								 | 
							
										return
							 | 
						||
| 
								 | 
							
									tblist = []
							 | 
						||
| 
								 | 
							
									while tb:
							 | 
						||
| 
								 | 
							
										tblist.append(tb)
							 | 
						||
| 
								 | 
							
										tb = tb.tb_next
							 | 
						||
| 
								 | 
							
									ptr = len(tblist)-1
							 | 
						||
| 
								 | 
							
									tb = tblist[ptr]
							 | 
						||
| 
								 | 
							
									while 1:
							 | 
						||
| 
								 | 
							
										if tb <> tblist[ptr]:
							 | 
						||
| 
								 | 
							
											tb = tblist[ptr]
							 | 
						||
| 
								 | 
							
											print `ptr` + ':',
							 | 
						||
| 
								 | 
							
											printtbheader(tb)
							 | 
						||
| 
								 | 
							
										try:
							 | 
						||
| 
								 | 
							
											line = raw_input('TB: ')
							 | 
						||
| 
								 | 
							
										except KeyboardInterrupt:
							 | 
						||
| 
								 | 
							
											print '\n[Interrupted]'
							 | 
						||
| 
								 | 
							
											break
							 | 
						||
| 
								 | 
							
										except EOFError:
							 | 
						||
| 
								 | 
							
											print '\n[EOF]'
							 | 
						||
| 
								 | 
							
											break
							 | 
						||
| 
								 | 
							
										cmd = string.strip(line)
							 | 
						||
| 
								 | 
							
										if cmd:
							 | 
						||
| 
								 | 
							
											if cmd = 'quit':
							 | 
						||
| 
								 | 
							
												break
							 | 
						||
| 
								 | 
							
											elif cmd = 'list':
							 | 
						||
| 
								 | 
							
												browserlist(tb)
							 | 
						||
| 
								 | 
							
											elif cmd = 'up':
							 | 
						||
| 
								 | 
							
												if ptr-1 >= 0: ptr = ptr-1
							 | 
						||
| 
								 | 
							
												else: print 'Bottom of stack.'
							 | 
						||
| 
								 | 
							
											elif cmd = 'down':
							 | 
						||
| 
								 | 
							
												if ptr+1 < len(tblist): ptr = ptr+1
							 | 
						||
| 
								 | 
							
												else: print 'Top of stack.'
							 | 
						||
| 
								 | 
							
											elif cmd = 'locals':
							 | 
						||
| 
								 | 
							
												printsymbols(tb.tb_frame.f_locals)
							 | 
						||
| 
								 | 
							
											elif cmd = 'globals':
							 | 
						||
| 
								 | 
							
												printsymbols(tb.tb_frame.f_globals)
							 | 
						||
| 
								 | 
							
											elif cmd in ('?', 'help'):
							 | 
						||
| 
								 | 
							
												browserhelp()
							 | 
						||
| 
								 | 
							
											else:
							 | 
						||
| 
								 | 
							
												browserexec(tb, cmd)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def browserlist(tb):
							 | 
						||
| 
								 | 
							
									filename = tb.tb_frame.f_code.co_filename
							 | 
						||
| 
								 | 
							
									lineno = tb.tb_lineno
							 | 
						||
| 
								 | 
							
									last = lineno
							 | 
						||
| 
								 | 
							
									first = max(1, last-10)
							 | 
						||
| 
								 | 
							
									for i in range(first, last+1):
							 | 
						||
| 
								 | 
							
										if i = lineno: prefix = '***' + string.rjust(`i`, 4) + ':'
							 | 
						||
| 
								 | 
							
										else: prefix = string.rjust(`i`, 7) + ':'
							 | 
						||
| 
								 | 
							
										line = readfileline(filename, i)
							 | 
						||
| 
								 | 
							
										if line[-1:] = '\n': line = line[:-1]
							 | 
						||
| 
								 | 
							
										print prefix + line
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def browserexec(tb, cmd):
							 | 
						||
| 
								 | 
							
									locals = tb.tb_frame.f_locals
							 | 
						||
| 
								 | 
							
									globals = tb.tb_frame.f_globals
							 | 
						||
| 
								 | 
							
									try:
							 | 
						||
| 
								 | 
							
										exec(cmd+'\n', globals, locals)
							 | 
						||
| 
								 | 
							
									except:
							 | 
						||
| 
								 | 
							
										print '*** Exception:',
							 | 
						||
| 
								 | 
							
										print sys.exc_type,
							 | 
						||
| 
								 | 
							
										if sys.exc_value <> None:
							 | 
						||
| 
								 | 
							
											print ':', sys.exc_value,
							 | 
						||
| 
								 | 
							
										print
							 | 
						||
| 
								 | 
							
										print 'Type help to get help.'
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def browserhelp():
							 | 
						||
| 
								 | 
							
									print
							 | 
						||
| 
								 | 
							
									print '    This is the traceback browser.  Commands are:'
							 | 
						||
| 
								 | 
							
									print '        up      : move one level up in the call stack'
							 | 
						||
| 
								 | 
							
									print '        down    : move one level down in the call stack'
							 | 
						||
| 
								 | 
							
									print '        locals  : print all local variables at this level'
							 | 
						||
| 
								 | 
							
									print '        globals : print all global variables at this level'
							 | 
						||
| 
								 | 
							
									print '        list    : list source code around the failure'
							 | 
						||
| 
								 | 
							
									print '        help    : print help (what you are reading now)'
							 | 
						||
| 
								 | 
							
									print '        quit    : back to command interpreter'
							 | 
						||
| 
								 | 
							
									print '    Typing any other 1-line statement will execute it'
							 | 
						||
| 
								 | 
							
									print '    using the current level\'s symbol tables'
							 | 
						||
| 
								 | 
							
									print
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def printtb(tb):
							 | 
						||
| 
								 | 
							
									while tb:
							 | 
						||
| 
								 | 
							
										print1tb(tb)
							 | 
						||
| 
								 | 
							
										tb = tb.tb_next
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def print1tb(tb):
							 | 
						||
| 
								 | 
							
									printtbheader(tb)
							 | 
						||
| 
								 | 
							
									if tb.tb_frame.f_locals is not tb.tb_frame.f_globals:
							 | 
						||
| 
								 | 
							
										printsymbols(tb.tb_frame.f_locals)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def printtbheader(tb):
							 | 
						||
| 
								 | 
							
									filename = tb.tb_frame.f_code.co_filename
							 | 
						||
| 
								 | 
							
									lineno = tb.tb_lineno
							 | 
						||
| 
								 | 
							
									info = '"' + filename + '"(' + `lineno` + ')'
							 | 
						||
| 
								 | 
							
									line = readfileline(filename, lineno)
							 | 
						||
| 
								 | 
							
									if line:
							 | 
						||
| 
								 | 
							
										info = info + ': ' + string.strip(line)
							 | 
						||
| 
								 | 
							
									print info
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def printsymbols(d):
							 | 
						||
| 
								 | 
							
									keys = d.keys()
							 | 
						||
| 
								 | 
							
									keys.sort()
							 | 
						||
| 
								 | 
							
									for name in keys:
							 | 
						||
| 
								 | 
							
										print '  ' + string.ljust(name, 12) + ':',
							 | 
						||
| 
								 | 
							
										printobject(d[name], 4)
							 | 
						||
| 
								 | 
							
										print
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def printobject(v, maxlevel):
							 | 
						||
| 
								 | 
							
									if v = None:
							 | 
						||
| 
								 | 
							
										print 'None',
							 | 
						||
| 
								 | 
							
									elif type(v) in (type(0), type(0.0)):
							 | 
						||
| 
								 | 
							
										print v,
							 | 
						||
| 
								 | 
							
									elif type(v) = type(''):
							 | 
						||
| 
								 | 
							
										if len(v) > 20:
							 | 
						||
| 
								 | 
							
											print `v[:17] + '...'`,
							 | 
						||
| 
								 | 
							
										else:
							 | 
						||
| 
								 | 
							
											print `v`,
							 | 
						||
| 
								 | 
							
									elif type(v) = type(()):
							 | 
						||
| 
								 | 
							
										print '(',
							 | 
						||
| 
								 | 
							
										printlist(v, maxlevel)
							 | 
						||
| 
								 | 
							
										print ')',
							 | 
						||
| 
								 | 
							
									elif type(v) = type([]):
							 | 
						||
| 
								 | 
							
										print '[',
							 | 
						||
| 
								 | 
							
										printlist(v, maxlevel)
							 | 
						||
| 
								 | 
							
										print ']',
							 | 
						||
| 
								 | 
							
									elif type(v) = type({}):
							 | 
						||
| 
								 | 
							
										print '{',
							 | 
						||
| 
								 | 
							
										printdict(v, maxlevel)
							 | 
						||
| 
								 | 
							
										print '}',
							 | 
						||
| 
								 | 
							
									else:
							 | 
						||
| 
								 | 
							
										print v,
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def printlist(v, maxlevel):
							 | 
						||
| 
								 | 
							
									n = len(v)
							 | 
						||
| 
								 | 
							
									if n = 0: return
							 | 
						||
| 
								 | 
							
									if maxlevel <= 0:
							 | 
						||
| 
								 | 
							
										print '...',
							 | 
						||
| 
								 | 
							
										return
							 | 
						||
| 
								 | 
							
									for i in range(min(6, n)):
							 | 
						||
| 
								 | 
							
										printobject(v[i], maxlevel-1)
							 | 
						||
| 
								 | 
							
										if i+1 < n: print ',',
							 | 
						||
| 
								 | 
							
									if n > 6: print '...',
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def printdict(v, maxlevel):
							 | 
						||
| 
								 | 
							
									keys = v.keys()
							 | 
						||
| 
								 | 
							
									n = len(keys)
							 | 
						||
| 
								 | 
							
									if n = 0: return
							 | 
						||
| 
								 | 
							
									if maxlevel <= 0:
							 | 
						||
| 
								 | 
							
										print '...',
							 | 
						||
| 
								 | 
							
										return
							 | 
						||
| 
								 | 
							
									keys.sort()
							 | 
						||
| 
								 | 
							
									for i in range(min(6, n)):
							 | 
						||
| 
								 | 
							
										key = keys[i]
							 | 
						||
| 
								 | 
							
										print `key` + ':',
							 | 
						||
| 
								 | 
							
										printobject(v[key], maxlevel-1)
							 | 
						||
| 
								 | 
							
										if i+1 < n: print ',',
							 | 
						||
| 
								 | 
							
									if n > 6: print '...',
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								_filecache = {}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def readfileline(filename, lineno):
							 | 
						||
| 
								 | 
							
									try:
							 | 
						||
| 
								 | 
							
										stat = os.stat(filename)
							 | 
						||
| 
								 | 
							
									except os.error, msg:
							 | 
						||
| 
								 | 
							
										print 'Cannot stat', filename, '--', msg
							 | 
						||
| 
								 | 
							
										return ''
							 | 
						||
| 
								 | 
							
									cache_ok = 0
							 | 
						||
| 
								 | 
							
									if _filecache.has_key(filename):
							 | 
						||
| 
								 | 
							
										cached_stat, lines = _filecache[filename]
							 | 
						||
| 
								 | 
							
										if stat[ST_SIZE] = cached_stat[ST_SIZE] and \
							 | 
						||
| 
								 | 
							
												stat[ST_MTIME] = cached_stat[ST_MTIME]:
							 | 
						||
| 
								 | 
							
											cache_ok = 1
							 | 
						||
| 
								 | 
							
										else:
							 | 
						||
| 
								 | 
							
											print 'Stale cache entry for', filename
							 | 
						||
| 
								 | 
							
											del _filecache[filename]
							 | 
						||
| 
								 | 
							
									if not cache_ok:
							 | 
						||
| 
								 | 
							
										lines = readfilelines(filename)
							 | 
						||
| 
								 | 
							
										if not lines:
							 | 
						||
| 
								 | 
							
											return ''
							 | 
						||
| 
								 | 
							
										_filecache[filename] = stat, lines
							 | 
						||
| 
								 | 
							
									if 0 <= lineno-1 < len(lines):
							 | 
						||
| 
								 | 
							
										return lines[lineno-1]
							 | 
						||
| 
								 | 
							
									else:
							 | 
						||
| 
								 | 
							
										print 'Line number out of range, last line is', len(lines)
							 | 
						||
| 
								 | 
							
										return ''
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def readfilelines(filename):
							 | 
						||
| 
								 | 
							
									try:
							 | 
						||
| 
								 | 
							
										fp = open(filename, 'r')
							 | 
						||
| 
								 | 
							
									except:
							 | 
						||
| 
								 | 
							
										print 'Cannot open', filename
							 | 
						||
| 
								 | 
							
										return []
							 | 
						||
| 
								 | 
							
									lines = []
							 | 
						||
| 
								 | 
							
									while 1:
							 | 
						||
| 
								 | 
							
										line = fp.readline()
							 | 
						||
| 
								 | 
							
										if not line: break
							 | 
						||
| 
								 | 
							
										lines.append(line)
							 | 
						||
| 
								 | 
							
									if not lines:
							 | 
						||
| 
								 | 
							
										print 'Empty file', filename
							 | 
						||
| 
								 | 
							
									return lines
							 |