mirror of https://github.com/python/cpython
Merged revisions 83523 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/branches/py3k ........ r83523 | georg.brandl | 2010-08-02 14:06:18 +0200 (Mo, 02 Aug 2010) | 1 line #9209 and #7781: fix two crashes in pstats interactive browser. ........
This commit is contained in:
parent
aa885af2d8
commit
9c5efadfb7
|
@ -5,7 +5,7 @@
|
||||||
# Based on prior profile module by Sjoerd Mullender...
|
# Based on prior profile module by Sjoerd Mullender...
|
||||||
# which was hacked somewhat by: Guido van Rossum
|
# which was hacked somewhat by: Guido van Rossum
|
||||||
#
|
#
|
||||||
# see profile.doc and profile.py for more info.
|
# see profile.py for more info.
|
||||||
|
|
||||||
# Copyright 1994, by InfoSeek Corporation, all rights reserved.
|
# Copyright 1994, by InfoSeek Corporation, all rights reserved.
|
||||||
# Written by James Roskind
|
# Written by James Roskind
|
||||||
|
@ -66,7 +66,7 @@ class Stats:
|
||||||
minor key of 'the name of the function'. Look at the two tables in
|
minor key of 'the name of the function'. Look at the two tables in
|
||||||
sort_stats() and get_sort_arg_defs(self) for more examples.
|
sort_stats() and get_sort_arg_defs(self) for more examples.
|
||||||
|
|
||||||
All methods return self, so you can string together commands like:
|
All methods return self, so you can string together commands like:
|
||||||
Stats('foo', 'goo').strip_dirs().sort_stats('calls').\
|
Stats('foo', 'goo').strip_dirs().sort_stats('calls').\
|
||||||
print_stats(5).print_callers(5)
|
print_stats(5).print_callers(5)
|
||||||
"""
|
"""
|
||||||
|
@ -150,7 +150,7 @@ class Stats:
|
||||||
if not arg_list: return self
|
if not arg_list: return self
|
||||||
if len(arg_list) > 1: self.add(*arg_list[1:])
|
if len(arg_list) > 1: self.add(*arg_list[1:])
|
||||||
other = arg_list[0]
|
other = arg_list[0]
|
||||||
if type(self) != type(other) or self.__class__ != other.__class__:
|
if type(self) != type(other):
|
||||||
other = Stats(other)
|
other = Stats(other)
|
||||||
self.files += other.files
|
self.files += other.files
|
||||||
self.total_calls += other.total_calls
|
self.total_calls += other.total_calls
|
||||||
|
@ -218,12 +218,12 @@ class Stats:
|
||||||
if not field:
|
if not field:
|
||||||
self.fcn_list = 0
|
self.fcn_list = 0
|
||||||
return self
|
return self
|
||||||
if len(field) == 1 and type(field[0]) == type(1):
|
if len(field) == 1 and isinstance(field[0], int):
|
||||||
# Be compatible with old profiler
|
# Be compatible with old profiler
|
||||||
field = [ {-1: "stdname",
|
field = [ {-1: "stdname",
|
||||||
0:"calls",
|
0: "calls",
|
||||||
1:"time",
|
1: "time",
|
||||||
2: "cumulative" } [ field[0] ] ]
|
2: "cumulative"}[field[0]] ]
|
||||||
|
|
||||||
sort_arg_defs = self.get_sort_arg_defs()
|
sort_arg_defs = self.get_sort_arg_defs()
|
||||||
sort_tuple = ()
|
sort_tuple = ()
|
||||||
|
@ -300,48 +300,53 @@ class Stats:
|
||||||
|
|
||||||
def eval_print_amount(self, sel, list, msg):
|
def eval_print_amount(self, sel, list, msg):
|
||||||
new_list = list
|
new_list = list
|
||||||
if type(sel) == type(""):
|
if isinstance(sel, str):
|
||||||
|
try:
|
||||||
|
rex = re.compile(sel)
|
||||||
|
except re.error:
|
||||||
|
msg += " <Invalid regular expression %r>\n" % sel
|
||||||
|
return new_list, msg
|
||||||
new_list = []
|
new_list = []
|
||||||
for func in list:
|
for func in list:
|
||||||
if re.search(sel, func_std_string(func)):
|
if rex.search(func_std_string(func)):
|
||||||
new_list.append(func)
|
new_list.append(func)
|
||||||
else:
|
else:
|
||||||
count = len(list)
|
count = len(list)
|
||||||
if type(sel) == type(1.0) and 0.0 <= sel < 1.0:
|
if isinstance(sel, float) and 0.0 <= sel < 1.0:
|
||||||
count = int(count * sel + .5)
|
count = int(count * sel + .5)
|
||||||
new_list = list[:count]
|
new_list = list[:count]
|
||||||
elif type(sel) == type(1) and 0 <= sel < count:
|
elif isinstance(sel, int) and 0 <= sel < count:
|
||||||
count = sel
|
count = sel
|
||||||
new_list = list[:count]
|
new_list = list[:count]
|
||||||
if len(list) != len(new_list):
|
if len(list) != len(new_list):
|
||||||
msg = msg + " List reduced from %r to %r due to restriction <%r>\n" % (
|
msg += " List reduced from %r to %r due to restriction <%r>\n" % (
|
||||||
len(list), len(new_list), sel)
|
len(list), len(new_list), sel)
|
||||||
|
|
||||||
return new_list, msg
|
return new_list, msg
|
||||||
|
|
||||||
def get_print_list(self, sel_list):
|
def get_print_list(self, sel_list):
|
||||||
width = self.max_name_len
|
width = self.max_name_len
|
||||||
if self.fcn_list:
|
if self.fcn_list:
|
||||||
list = self.fcn_list[:]
|
stat_list = self.fcn_list[:]
|
||||||
msg = " Ordered by: " + self.sort_type + '\n'
|
msg = " Ordered by: " + self.sort_type + '\n'
|
||||||
else:
|
else:
|
||||||
list = self.stats.keys()
|
stat_list = list(self.stats.keys())
|
||||||
msg = " Random listing order was used\n"
|
msg = " Random listing order was used\n"
|
||||||
|
|
||||||
for selection in sel_list:
|
for selection in sel_list:
|
||||||
list, msg = self.eval_print_amount(selection, list, msg)
|
stat_list, msg = self.eval_print_amount(selection, stat_list, msg)
|
||||||
|
|
||||||
count = len(list)
|
count = len(stat_list)
|
||||||
|
|
||||||
if not list:
|
if not stat_list:
|
||||||
return 0, list
|
return 0, stat_list
|
||||||
print >> self.stream, msg
|
print >> self.stream, msg
|
||||||
if count < len(self.stats):
|
if count < len(self.stats):
|
||||||
width = 0
|
width = 0
|
||||||
for func in list:
|
for func in stat_list:
|
||||||
if len(func_std_string(func)) > width:
|
if len(func_std_string(func)) > width:
|
||||||
width = len(func_std_string(func))
|
width = len(func_std_string(func))
|
||||||
return width+2, list
|
return width+2, stat_list
|
||||||
|
|
||||||
def print_stats(self, *amount):
|
def print_stats(self, *amount):
|
||||||
for filename in self.files:
|
for filename in self.files:
|
||||||
|
@ -553,12 +558,10 @@ if __name__ == '__main__':
|
||||||
def __init__(self, profile=None):
|
def __init__(self, profile=None):
|
||||||
cmd.Cmd.__init__(self)
|
cmd.Cmd.__init__(self)
|
||||||
self.prompt = "% "
|
self.prompt = "% "
|
||||||
|
self.stats = None
|
||||||
|
self.stream = sys.stdout
|
||||||
if profile is not None:
|
if profile is not None:
|
||||||
self.stats = Stats(profile)
|
self.do_read(profile)
|
||||||
self.stream = self.stats.stream
|
|
||||||
else:
|
|
||||||
self.stats = None
|
|
||||||
self.stream = sys.stdout
|
|
||||||
|
|
||||||
def generic(self, fn, line):
|
def generic(self, fn, line):
|
||||||
args = line.split()
|
args = line.split()
|
||||||
|
|
|
@ -26,6 +26,12 @@ Library
|
||||||
|
|
||||||
- Issue #9354: Provide getsockopt() in asyncore's file_wrapper.
|
- Issue #9354: Provide getsockopt() in asyncore's file_wrapper.
|
||||||
|
|
||||||
|
- Issue #7781: Fix restricting stats by entry counts in the pstats
|
||||||
|
interactive browser.
|
||||||
|
|
||||||
|
- Issue #9209: Do not crash in the pstats interactive browser on invalid
|
||||||
|
regular expressions.
|
||||||
|
|
||||||
- Issue #7372: Fix pstats regression when stripping paths from profile
|
- Issue #7372: Fix pstats regression when stripping paths from profile
|
||||||
data generated with the profile module.
|
data generated with the profile module.
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue