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