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