Trace.py 4.04 KB
Newer Older
1
"""Tracing metaclass.
2

3 4 5 6 7
XXX This is very much a work in progress.

"""

import types, sys
8 9 10 11 12 13 14 15 16 17 18 19 20 21 22

class TraceMetaClass:
    """Metaclass for tracing.

    Classes defined using this metaclass have an automatic tracing
    feature -- by setting the __trace_output__ instance (or class)
    variable to a file object, trace messages about all calls are
    written to the file.  The trace formatting can be changed by
    defining a suitable __trace_call__ method.

    """

    __inited = 0

    def __init__(self, name, bases, dict):
Guido van Rossum's avatar
Guido van Rossum committed
23 24 25 26 27
        self.__name__ = name
        self.__bases__ = bases
        self.__dict = dict
        # XXX Can't define __dict__, alas
        self.__inited = 1
28 29

    def __getattr__(self, name):
Guido van Rossum's avatar
Guido van Rossum committed
30 31 32 33 34 35 36 37 38
        try:
            return self.__dict[name]
        except KeyError:
            for base in self.__bases__:
                try:
                    return base.__getattr__(name)
                except AttributeError:
                    pass
            raise AttributeError, name
39 40

    def __setattr__(self, name, value):
Guido van Rossum's avatar
Guido van Rossum committed
41 42 43 44
        if not self.__inited:
            self.__dict__[name] = value
        else:
            self.__dict[name] = value
45 46

    def __call__(self, *args, **kw):
Guido van Rossum's avatar
Guido van Rossum committed
47 48 49 50 51 52 53 54
        inst = TracingInstance()
        inst.__meta_init__(self)
        try:
            init = inst.__getattr__('__init__')
        except AttributeError:
            init = lambda: None
        apply(init, args, kw)
        return inst
55 56 57 58 59 60 61

    __trace_output__ = None

class TracingInstance:
    """Helper class to represent an instance of a tracing class."""

    def __trace_call__(self, fp, fmt, *args):
Guido van Rossum's avatar
Guido van Rossum committed
62
        fp.write((fmt+'\n') % args)
63 64

    def __meta_init__(self, klass):
Guido van Rossum's avatar
Guido van Rossum committed
65
        self.__class = klass
66 67

    def __getattr__(self, name):
Guido van Rossum's avatar
Guido van Rossum committed
68 69 70 71 72 73 74 75 76 77 78 79 80
        # Invoked for any attr not in the instance's __dict__
        try:
            raw = self.__class.__getattr__(name)
        except AttributeError:
            raise AttributeError, name
        if type(raw) != types.FunctionType:
            return raw
        # It's a function
        fullname = self.__class.__name__ + "." + name
        if not self.__trace_output__ or name == '__trace_call__':
            return NotTracingWrapper(fullname, raw, self)
        else:
            return TracingWrapper(fullname, raw, self)
81 82 83

class NotTracingWrapper:
    def __init__(self, name, func, inst):
Guido van Rossum's avatar
Guido van Rossum committed
84 85 86
        self.__name__ = name
        self.func = func
        self.inst = inst
87
    def __call__(self, *args, **kw):
Guido van Rossum's avatar
Guido van Rossum committed
88
        return apply(self.func, (self.inst,) + args, kw)
89 90 91

class TracingWrapper(NotTracingWrapper):
    def __call__(self, *args, **kw):
Guido van Rossum's avatar
Guido van Rossum committed
92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
        self.inst.__trace_call__(self.inst.__trace_output__,
                                 "calling %s, inst=%s, args=%s, kw=%s",
                                 self.__name__, self.inst, args, kw)
        try:
            rv = apply(self.func, (self.inst,) + args, kw)
        except:
            t, v, tb = sys.exc_info()
            self.inst.__trace_call__(self.inst.__trace_output__,
                                     "returning from %s with exception %s: %s",
                                     self.__name__, t, v)
            raise t, v, tb
        else:
            self.inst.__trace_call__(self.inst.__trace_output__,
                                     "returning from %s with value %s",
                                     self.__name__, rv)
            return rv
108 109 110 111 112

Traced = TraceMetaClass('Traced', (), {'__trace_output__': None})


def _test():
113
    global C, D
114
    class C(Traced):
Guido van Rossum's avatar
Guido van Rossum committed
115 116 117 118
        def __init__(self, x=0): self.x = x
        def m1(self, x): self.x = x
        def m2(self, y): return self.x + y
        __trace_output__ = sys.stdout
119
    class D(C):
Guido van Rossum's avatar
Guido van Rossum committed
120 121
        def m2(self, y): print "D.m2(%s)" % `y`; return C.m2(self, y)
        __trace_output__ = None
122 123 124 125 126 127 128 129 130 131
    x = C(4321)
    print x
    print x.x
    print x.m1(100)
    print x.m1(10)
    print x.m2(33)
    print x.m1(5)
    print x.m2(4000)
    print x.x

132 133 134 135 136 137 138 139 140 141 142
    print C.__init__
    print C.m2
    print D.__init__
    print D.m2

    y = D()
    print y
    print y.m1(10)
    print y.m2(100)
    print y.x

143 144
if __name__ == '__main__':
    _test()
145