/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
1185.33.86 by Martin Pool
Add additional module for lsprof.
1
# this is copied from the lsprof distro because somehow
2
# it is not installed by distutils
3
# I made one modification to profile so that it returns a pair
4
# instead of just the Stats object
5
6379.6.3 by Jelmer Vernooij
Use absolute_import.
6
from __future__ import absolute_import
7
6816.3.3 by Martin
Wrap lsprof text oriented file output with a UTF-8 encoder
8
import codecs
6816.3.1 by Martin
Make lsprof import on Python 3 but not pass all tests yet
9
try:
10
    import cPickle as pickle
11
except ImportError:
12
    import pickle
13
import operator
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
14
import os
1185.33.86 by Martin Pool
Add additional module for lsprof.
15
import sys
6816.3.1 by Martin
Make lsprof import on Python 3 but not pass all tests yet
16
try:
17
    import _thread
18
except ImportError:
19
    import thread as _thread
1553.7.2 by Robey Pointer
add threading support -- each thread created during an lsprof session gets its own profile object, and the Stats objects are attached to a 'threads' member of the final Stats object
20
import threading
1773.4.1 by Martin Pool
Add pyflakes makefile target; fix many warnings
21
from _lsprof import Profiler, profiler_entry
1185.33.86 by Martin Pool
Add additional module for lsprof.
22
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
23
from . import errors
2805.7.4 by Ian Clatworthy
incorporate feedback from lifeless
24
1185.33.86 by Martin Pool
Add additional module for lsprof.
25
__all__ = ['profile', 'Stats']
26
7143.15.2 by Jelmer Vernooij
Run autopep8.
27
1185.33.86 by Martin Pool
Add additional module for lsprof.
28
def profile(f, *args, **kwds):
3696.3.5 by Robert Collins
Streamline _walkdirs_utf8 for utf8 file systems, reducing time to traverse a mozilla tree from 1s to .6 seconds. (Robert Collins)
29
    """Run a function profile.
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
30
4084.6.1 by Robert Collins
Refactor profiling exception handling to restore clear layers - command handling in commands.py, profiling in lsprof.py.
31
    Exceptions are not caught: If you need stats even when exceptions are to be
4641.3.2 by Robert Collins
lsprof support.
32
    raised, pass in a closure that will catch the exceptions and transform them
33
    appropriately for your driver function.
4084.6.1 by Robert Collins
Refactor profiling exception handling to restore clear layers - command handling in commands.py, profiling in lsprof.py.
34
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
35
    Important caveat: only one profile can execute at a time. See BzrProfiler
36
    for details.
37
3696.3.5 by Robert Collins
Streamline _walkdirs_utf8 for utf8 file systems, reducing time to traverse a mozilla tree from 1s to .6 seconds. (Robert Collins)
38
    :return: The functions return value and a stats object.
39
    """
4641.3.2 by Robert Collins
lsprof support.
40
    profiler = BzrProfiler()
41
    profiler.start()
1185.33.86 by Martin Pool
Add additional module for lsprof.
42
    try:
4084.6.1 by Robert Collins
Refactor profiling exception handling to restore clear layers - command handling in commands.py, profiling in lsprof.py.
43
        ret = f(*args, **kwds)
1185.33.86 by Martin Pool
Add additional module for lsprof.
44
    finally:
4641.3.2 by Robert Collins
lsprof support.
45
        stats = profiler.stop()
46
    return ret, stats
47
48
49
class BzrProfiler(object):
50
    """Bzr utility wrapper around Profiler.
7143.15.2 by Jelmer Vernooij
Run autopep8.
51
4641.3.2 by Robert Collins
lsprof support.
52
    For most uses the module level 'profile()' function will be suitable.
53
    However profiling when a simple wrapped function isn't available may
54
    be easier to accomplish using this class.
55
56
    To use it, create a BzrProfiler and call start() on it. Some arbitrary
57
    time later call stop() to stop profiling and retrieve the statistics
58
    from the code executed in the interim.
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
59
60
    Note that profiling involves a threading.Lock around the actual profiling.
61
    This is needed because profiling involves global manipulation of the python
62
    interpreter state. As such you cannot perform multiple profiles at once.
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
63
    Trying to do so will lock out the second profiler unless the global
6622.1.34 by Jelmer Vernooij
Rename brzlib => breezy.
64
    breezy.lsprof.BzrProfiler.profiler_block is set to 0. Setting it to 0 will
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
65
    cause profiling to fail rather than blocking.
4641.3.2 by Robert Collins
lsprof support.
66
    """
67
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
68
    profiler_block = 1
69
    """Serialise rather than failing to profile concurrent profile requests."""
70
71
    profiler_lock = threading.Lock()
72
    """Global lock used to serialise profiles."""
73
4641.3.2 by Robert Collins
lsprof support.
74
    def start(self):
75
        """Start profiling.
7143.15.2 by Jelmer Vernooij
Run autopep8.
76
4641.3.2 by Robert Collins
lsprof support.
77
        This hooks into threading and will record all calls made until
78
        stop() is called.
79
        """
80
        self._g_threadmap = {}
81
        self.p = Profiler()
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
82
        permitted = self.__class__.profiler_lock.acquire(
83
            self.__class__.profiler_block)
84
        if not permitted:
85
            raise errors.InternalBzrError(msg="Already profiling something")
86
        try:
87
            self.p.enable(subcalls=True)
88
            threading.setprofile(self._thread_profile)
7143.15.5 by Jelmer Vernooij
More PEP8 fixes.
89
        except BaseException:
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
90
            self.__class__.profiler_lock.release()
91
            raise
4641.3.2 by Robert Collins
lsprof support.
92
93
    def stop(self):
94
        """Stop profiling.
95
96
        This unhooks from threading and cleans up the profiler, returning
97
        the gathered Stats object.
98
6622.1.34 by Jelmer Vernooij
Rename brzlib => breezy.
99
        :return: A breezy.lsprof.Stats object.
4641.3.2 by Robert Collins
lsprof support.
100
        """
5331.1.1 by Robert Collins
``bzrlib.lsprof.profile`` will no longer silently generate bad threaded
101
        try:
102
            self.p.disable()
103
            for pp in self._g_threadmap.values():
104
                pp.disable()
105
            threading.setprofile(None)
106
            p = self.p
107
            self.p = None
108
            threads = {}
109
            for tid, pp in self._g_threadmap.items():
110
                threads[tid] = Stats(pp.getstats(), {})
111
            self._g_threadmap = None
112
            return Stats(p.getstats(), threads)
113
        finally:
114
            self.__class__.profiler_lock.release()
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
115
4641.3.2 by Robert Collins
lsprof support.
116
    def _thread_profile(self, f, *args, **kwds):
117
        # we lose the first profile point for a new thread in order to
118
        # trampoline a new Profile object into place
6816.3.1 by Martin
Make lsprof import on Python 3 but not pass all tests yet
119
        thr = _thread.get_ident()
4641.3.2 by Robert Collins
lsprof support.
120
        self._g_threadmap[thr] = p = Profiler()
121
        # this overrides our sys.setprofile hook:
122
        p.enable(subcalls=True, builtins=True)
1185.33.86 by Martin Pool
Add additional module for lsprof.
123
124
125
class Stats(object):
5432.2.1 by John C Barstow
Provide missing docstrings for bzrlib.lsprof
126
    """Wrapper around the collected data.
127
128
    A Stats instance is created when the profiler finishes. Normal
129
    usage is to use save() to write out the data to a file, or pprint()
130
    to write human-readable information to the command line.
131
    """
1185.33.86 by Martin Pool
Add additional module for lsprof.
132
1553.7.2 by Robey Pointer
add threading support -- each thread created during an lsprof session gets its own profile object, and the Stats objects are attached to a 'threads' member of the final Stats object
133
    def __init__(self, data, threads):
1185.33.86 by Martin Pool
Add additional module for lsprof.
134
        self.data = data
1553.7.2 by Robey Pointer
add threading support -- each thread created during an lsprof session gets its own profile object, and the Stats objects are attached to a 'threads' member of the final Stats object
135
        self.threads = threads
1185.33.86 by Martin Pool
Add additional module for lsprof.
136
6816.3.2 by Martin
Test lsprof stat sorting and make pass on Python 3
137
    def sort(self, crit="inlinetime", reverse=True):
5432.2.1 by John C Barstow
Provide missing docstrings for bzrlib.lsprof
138
        """Sort the data by the supplied critera.
139
140
        :param crit: the data attribute used as the sort key."""
6816.3.2 by Martin
Test lsprof stat sorting and make pass on Python 3
141
        if crit not in profiler_entry.__dict__ or crit == 'code':
6619.3.11 by Jelmer Vernooij
Use modern exceptions.
142
            raise ValueError("Can't sort by %s" % crit)
6816.3.2 by Martin
Test lsprof stat sorting and make pass on Python 3
143
144
        key_func = operator.attrgetter(crit)
145
        self.data.sort(key=key_func, reverse=reverse)
146
1185.33.86 by Martin Pool
Add additional module for lsprof.
147
        for e in self.data:
148
            if e.calls:
6816.3.2 by Martin
Test lsprof stat sorting and make pass on Python 3
149
                e.calls.sort(key=key_func, reverse=reverse)
1185.33.86 by Martin Pool
Add additional module for lsprof.
150
151
    def pprint(self, top=None, file=None):
5432.2.1 by John C Barstow
Provide missing docstrings for bzrlib.lsprof
152
        """Pretty-print the data as plain text for human consumption.
153
154
        :param top: only output the top n entries.
155
            The default value of None means output all data.
156
        :param file: the output file; if None, output will
157
            default to stdout."""
1185.33.86 by Martin Pool
Add additional module for lsprof.
158
        if file is None:
159
            file = sys.stdout
160
        d = self.data
161
        if top is not None:
162
            d = d[:top]
163
        cols = "% 12s %12s %11.4f %11.4f   %s\n"
164
        hcols = "% 12s %12s %12s %12s %s\n"
165
        file.write(hcols % ("CallCount", "Recursive", "Total(ms)",
166
                            "Inline(ms)", "module:lineno(function)"))
167
        for e in d:
168
            file.write(cols % (e.callcount, e.reccallcount, e.totaltime,
169
                               e.inlinetime, label(e.code)))
170
            if e.calls:
171
                for se in e.calls:
172
                    file.write(cols % ("+%s" % se.callcount, se.reccallcount,
173
                                       se.totaltime, se.inlinetime,
174
                                       "+%s" % label(se.code)))
175
176
    def freeze(self):
177
        """Replace all references to code objects with string
178
        descriptions; this makes it possible to pickle the instance."""
179
180
        # this code is probably rather ickier than it needs to be!
181
        for i in range(len(self.data)):
182
            e = self.data[i]
183
            if not isinstance(e.code, str):
184
                self.data[i] = type(e)((label(e.code),) + e[1:])
1706.2.7 by Robey Pointer
pull over some changes from the original lsprof
185
            if e.calls:
186
                for j in range(len(e.calls)):
187
                    se = e.calls[j]
188
                    if not isinstance(se.code, str):
189
                        e.calls[j] = type(se)((label(se.code),) + se[1:])
1553.7.2 by Robey Pointer
add threading support -- each thread created during an lsprof session gets its own profile object, and the Stats objects are attached to a 'threads' member of the final Stats object
190
        for s in self.threads.values():
191
            s.freeze()
192
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
193
    def calltree(self, file):
194
        """Output profiling data in calltree format (for KCacheGrind)."""
195
        _CallTreeFilter(self.data).output(file)
196
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
197
    def save(self, filename, format=None):
198
        """Save profiling data to a file.
199
200
        :param filename: the name of the output file
201
        :param format: 'txt' for a text representation;
202
            'callgrind' for calltree format;
203
            otherwise a pickled Python object. A format of None indicates
2654.2.2 by Ian Clatworthy
Put all format detection stuff in one spot as requested by John Arbash Meinel
204
            that the format to use is to be found from the filename. If
205
            the name starts with callgrind.out, callgrind format is used
206
            otherwise the format is given by the filename extension.
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
207
        """
208
        if format is None:
3535.5.1 by John Arbash Meinel
cleanup a few imports to be lazily loaded.
209
            basename = os.path.basename(filename)
2805.7.2 by Ian Clatworthy
use basename, not full path, when checking for callgrind.out file prefix
210
            if basename.startswith('callgrind.out'):
2654.2.2 by Ian Clatworthy
Put all format detection stuff in one spot as requested by John Arbash Meinel
211
                format = "callgrind"
212
            else:
3535.5.1 by John Arbash Meinel
cleanup a few imports to be lazily loaded.
213
                ext = os.path.splitext(filename)[1]
2654.2.2 by Ian Clatworthy
Put all format detection stuff in one spot as requested by John Arbash Meinel
214
                if len(ext) > 1:
215
                    format = ext[1:]
6816.3.1 by Martin
Make lsprof import on Python 3 but not pass all tests yet
216
        with open(filename, 'wb') as outfile:
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
217
            if format == "callgrind":
6816.3.3 by Martin
Wrap lsprof text oriented file output with a UTF-8 encoder
218
                # The callgrind format states it is 'ASCII based':
219
                # <http://valgrind.org/docs/manual/cl-format.html>
220
                # But includes filenames so lets ignore and use UTF-8.
221
                self.calltree(codecs.getwriter('utf-8')(outfile))
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
222
            elif format == "txt":
6816.3.3 by Martin
Wrap lsprof text oriented file output with a UTF-8 encoder
223
                self.pprint(file=codecs.getwriter('utf-8')(outfile))
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
224
            else:
225
                self.freeze()
6816.3.1 by Martin
Make lsprof import on Python 3 but not pass all tests yet
226
                pickle.dump(self, outfile, 2)
2493.2.3 by Ian Clatworthy
changes requested in jameinel's review incorporated
227
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
228
229
class _CallTreeFilter(object):
2493.2.2 by Ian Clatworthy
Incorporate feedback from Robert Collins
230
    """Converter of a Stats object to input suitable for KCacheGrind.
231
232
    This code is taken from http://ddaa.net/blog/python/lsprof-calltree
233
    with the changes made by J.P. Calderone and Itamar applied. Note that
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
234
    isinstance(code, str) needs to be used at times to determine if the code
2493.2.2 by Ian Clatworthy
Incorporate feedback from Robert Collins
235
    object is actually an external code object (with a filename, etc.) or
236
    a Python built-in.
237
    """
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
238
239
    def __init__(self, data):
240
        self.data = data
241
        self.out_file = None
242
243
    def output(self, out_file):
3943.8.1 by Marius Kruger
remove all trailing whitespace from bzr source
244
        self.out_file = out_file
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
245
        out_file.write('events: Ticks\n')
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
246
        self._print_summary()
247
        for entry in self.data:
248
            self._entry(entry)
249
250
    def _print_summary(self):
251
        max_cost = 0
252
        for entry in self.data:
253
            totaltime = int(entry.totaltime * 1000)
254
            max_cost = max(max_cost, totaltime)
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
255
        self.out_file.write('summary: %d\n' % (max_cost,))
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
256
257
    def _entry(self, entry):
258
        out_file = self.out_file
259
        code = entry.code
260
        inlinetime = int(entry.inlinetime * 1000)
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
261
        if isinstance(code, str):
262
            out_file.write('fi=~\n')
263
        else:
264
            out_file.write('fi=%s\n' % (code.co_filename,))
265
        out_file.write('fn=%s\n' % (label(code, True),))
266
        if isinstance(code, str):
2911.6.4 by Blake Winton
Fix test failures
267
            out_file.write('0  %s\n' % (inlinetime,))
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
268
        else:
269
            out_file.write('%d %d\n' % (code.co_firstlineno, inlinetime))
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
270
        # recursive calls are counted in entry.calls
271
        if entry.calls:
272
            calls = entry.calls
273
        else:
274
            calls = []
2493.2.1 by Ian Clatworthy
make profiling information easier to view and better documented
275
        if isinstance(code, str):
276
            lineno = 0
277
        else:
278
            lineno = code.co_firstlineno
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
279
        for subentry in calls:
2493.2.1 by Ian Clatworthy
make profiling information easier to view and better documented
280
            self._subentry(lineno, subentry)
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
281
        out_file.write('\n')
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
282
283
    def _subentry(self, lineno, subentry):
284
        out_file = self.out_file
285
        code = subentry.code
286
        totaltime = int(subentry.totaltime * 1000)
2493.2.1 by Ian Clatworthy
make profiling information easier to view and better documented
287
        if isinstance(code, str):
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
288
            out_file.write('cfi=~\n')
5783.1.1 by John Arbash Meinel
Fix bug #758695, correctly order the 'cfn' vs 'cfi' in callgrind output.
289
            out_file.write('cfn=%s\n' % (label(code, True),))
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
290
            out_file.write('calls=%d 0\n' % (subentry.callcount,))
2493.2.1 by Ian Clatworthy
make profiling information easier to view and better documented
291
        else:
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
292
            out_file.write('cfi=%s\n' % (code.co_filename,))
5783.1.1 by John Arbash Meinel
Fix bug #758695, correctly order the 'cfn' vs 'cfi' in callgrind output.
293
            out_file.write('cfn=%s\n' % (label(code, True),))
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
294
            out_file.write('calls=%d %d\n' % (
295
                subentry.callcount, code.co_firstlineno))
296
        out_file.write('%d %d\n' % (lineno, totaltime))
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
297
7143.15.2 by Jelmer Vernooij
Run autopep8.
298
1185.33.86 by Martin Pool
Add additional module for lsprof.
299
_fn2mod = {}
300
7143.15.2 by Jelmer Vernooij
Run autopep8.
301
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
302
def label(code, calltree=False):
1185.33.86 by Martin Pool
Add additional module for lsprof.
303
    if isinstance(code, str):
304
        return code
305
    try:
306
        mname = _fn2mod[code.co_filename]
307
    except KeyError:
1711.2.124 by John Arbash Meinel
Use sys.modules.items() to prevent running into site.py problems
308
        for k, v in sys.modules.items():
1185.33.86 by Martin Pool
Add additional module for lsprof.
309
            if v is None:
310
                continue
1963.2.6 by Robey Pointer
pychecker is on crack; go back to using 'is None'.
311
            if getattr(v, '__file__', None) is None:
1185.33.86 by Martin Pool
Add additional module for lsprof.
312
                continue
313
            if not isinstance(v.__file__, str):
314
                continue
315
            if v.__file__.startswith(code.co_filename):
316
                mname = _fn2mod[code.co_filename] = k
317
                break
318
        else:
7143.15.2 by Jelmer Vernooij
Run autopep8.
319
            mname = _fn2mod[code.co_filename] = '<%s>' % code.co_filename
1553.7.3 by Robey Pointer
patch from ddaa to add kcachegrind output-format support to lsprof
320
    if calltree:
321
        return '%s %s:%d' % (code.co_name, mname, code.co_firstlineno)
322
    else:
323
        return '%s:%d(%s)' % (mname, code.co_firstlineno, code.co_name)
1185.33.86 by Martin Pool
Add additional module for lsprof.
324
325
6816.3.4 by Martin
Create a cross compatible lsprof main()
326
def main():
1185.33.86 by Martin Pool
Add additional module for lsprof.
327
    sys.argv = sys.argv[1:]
328
    if not sys.argv:
2911.6.1 by Blake Winton
Change 'print >> f,'s to 'f.write('s.
329
        sys.stderr.write("usage: lsprof.py <script> <arguments...>\n")
1185.33.86 by Martin Pool
Add additional module for lsprof.
330
        sys.exit(2)
6816.3.4 by Martin
Create a cross compatible lsprof main()
331
    import runpy
332
    result, stats = profile(runpy.run_path, sys.argv[0], run_name='__main__')
333
    stats.sort()
1185.33.86 by Martin Pool
Add additional module for lsprof.
334
    stats.pprint()
6816.3.4 by Martin
Create a cross compatible lsprof main()
335
336
337
if __name__ == '__main__':
338
    main()