stackviewer.py 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. import linecache
  2. import os
  3. import sys
  4. import tkinter as tk
  5. from idlelib.debugobj import ObjectTreeItem, make_objecttreeitem
  6. from idlelib.tree import TreeNode, TreeItem, ScrolledCanvas
  7. def StackBrowser(root, flist=None, tb=None, top=None):
  8. global sc, item, node # For testing.
  9. if top is None:
  10. top = tk.Toplevel(root)
  11. sc = ScrolledCanvas(top, bg="white", highlightthickness=0)
  12. sc.frame.pack(expand=1, fill="both")
  13. item = StackTreeItem(flist, tb)
  14. node = TreeNode(sc.canvas, None, item)
  15. node.expand()
  16. class StackTreeItem(TreeItem):
  17. def __init__(self, flist=None, tb=None):
  18. self.flist = flist
  19. self.stack = self.get_stack(tb)
  20. self.text = self.get_exception()
  21. def get_stack(self, tb):
  22. if tb is None:
  23. tb = sys.last_traceback
  24. stack = []
  25. if tb and tb.tb_frame is None:
  26. tb = tb.tb_next
  27. while tb is not None:
  28. stack.append((tb.tb_frame, tb.tb_lineno))
  29. tb = tb.tb_next
  30. return stack
  31. def get_exception(self):
  32. type = sys.last_type
  33. value = sys.last_value
  34. if hasattr(type, "__name__"):
  35. type = type.__name__
  36. s = str(type)
  37. if value is not None:
  38. s = s + ": " + str(value)
  39. return s
  40. def GetText(self):
  41. return self.text
  42. def GetSubList(self):
  43. sublist = []
  44. for info in self.stack:
  45. item = FrameTreeItem(info, self.flist)
  46. sublist.append(item)
  47. return sublist
  48. class FrameTreeItem(TreeItem):
  49. def __init__(self, info, flist):
  50. self.info = info
  51. self.flist = flist
  52. def GetText(self):
  53. frame, lineno = self.info
  54. try:
  55. modname = frame.f_globals["__name__"]
  56. except:
  57. modname = "?"
  58. code = frame.f_code
  59. filename = code.co_filename
  60. funcname = code.co_name
  61. sourceline = linecache.getline(filename, lineno)
  62. sourceline = sourceline.strip()
  63. if funcname in ("?", "", None):
  64. item = "%s, line %d: %s" % (modname, lineno, sourceline)
  65. else:
  66. item = "%s.%s(...), line %d: %s" % (modname, funcname,
  67. lineno, sourceline)
  68. return item
  69. def GetSubList(self):
  70. frame, lineno = self.info
  71. sublist = []
  72. if frame.f_globals is not frame.f_locals:
  73. item = VariablesTreeItem("<locals>", frame.f_locals, self.flist)
  74. sublist.append(item)
  75. item = VariablesTreeItem("<globals>", frame.f_globals, self.flist)
  76. sublist.append(item)
  77. return sublist
  78. def OnDoubleClick(self):
  79. if self.flist:
  80. frame, lineno = self.info
  81. filename = frame.f_code.co_filename
  82. if os.path.isfile(filename):
  83. self.flist.gotofileline(filename, lineno)
  84. class VariablesTreeItem(ObjectTreeItem):
  85. def GetText(self):
  86. return self.labeltext
  87. def GetLabelText(self):
  88. return None
  89. def IsExpandable(self):
  90. return len(self.object) > 0
  91. def GetSubList(self):
  92. sublist = []
  93. for key in self.object.keys():
  94. try:
  95. value = self.object[key]
  96. except KeyError:
  97. continue
  98. def setfunction(value, key=key, object=self.object):
  99. object[key] = value
  100. item = make_objecttreeitem(key + " =", value, setfunction)
  101. sublist.append(item)
  102. return sublist
  103. def _stack_viewer(parent): # htest #
  104. from idlelib.pyshell import PyShellFileList
  105. top = tk.Toplevel(parent)
  106. top.title("Test StackViewer")
  107. x, y = map(int, parent.geometry().split('+')[1:])
  108. top.geometry("+%d+%d" % (x + 50, y + 175))
  109. flist = PyShellFileList(top)
  110. try: # to obtain a traceback object
  111. intentional_name_error
  112. except NameError:
  113. exc_type, exc_value, exc_tb = sys.exc_info()
  114. # inject stack trace to sys
  115. sys.last_type = exc_type
  116. sys.last_value = exc_value
  117. sys.last_traceback = exc_tb
  118. StackBrowser(top, flist=flist, top=top, tb=exc_tb)
  119. # restore sys to original state
  120. del sys.last_type
  121. del sys.last_value
  122. del sys.last_traceback
  123. if __name__ == '__main__':
  124. from unittest import main
  125. main('idlelib.idle_test.test_stackviewer', verbosity=2, exit=False)
  126. from idlelib.idle_test.htest import run
  127. run(_stack_viewer)