summaryrefslogtreecommitdiffstats
path: root/Demo/metaclasses/Meta.py
blob: 76193c1f3873acdecbe7873611e51f60fb68c43f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
"""Generic metaclass.

XXX This is very much a work in progress.

"""

import types

class MetaMethodWrapper:

    def __init__(self, func, inst):
	self.func = func
	self.inst = inst
	self.__name__ = self.func.__name__

    def __call__(self, *args, **kw):
	return apply(self.func, (self.inst,) + args, kw)

class MetaHelper:

    __methodwrapper__ = MetaMethodWrapper # For derived helpers to override

    def __helperinit__(self, formalclass):
	self.__formalclass__ = formalclass

    def __getattr__(self, name):
	# Invoked for any attr not in the instance's __dict__
	try:
	    raw = self.__formalclass__.__getattr__(name)
	except AttributeError:
	    try:
		ga = self.__formalclass__.__getattr__('__usergetattr__')
	    except KeyError:
		raise AttributeError, name
	    return ga(self, name)
	if type(raw) != types.FunctionType:
	    return raw
	return self.__methodwrapper__(raw, self)

class MetaClass:

    """A generic metaclass.

    This can be subclassed to implement various kinds of meta-behavior.

    """

    __helper__ = MetaHelper		# For derived metaclasses to override

    __inited = 0

    def __init__(self, name, bases, dict):
	try:
	    ga = dict['__getattr__']
	except KeyError:
	    pass
	else:
	    dict['__usergetattr__'] = ga
	    del dict['__getattr__']
	self.__name__ = name
	self.__bases__ = bases
	self.__realdict__ = dict
	self.__inited = 1

    def __getattr__(self, name):
	try:
	    return self.__realdict__[name]
	except KeyError:
	    for base in self.__bases__:
		try:
		    return base.__getattr__(name)
		except AttributeError:
		    pass
	    raise AttributeError, name

    def __setattr__(self, name, value):
	if not self.__inited:
	    self.__dict__[name] = value
	else:
	    self.__realdict__[name] = value

    def __call__(self, *args, **kw):
	inst = self.__helper__()
	inst.__helperinit__(self)
	try:
	    init = inst.__getattr__('__init__')
	except AttributeError:
	    init = lambda: None
	apply(init, args, kw)
	return inst
    

Meta = MetaClass('Meta', (), {})


def _test():
    class C(Meta):
	def __init__(self, *args):
	    print "__init__, args =", args
	def m1(self, x):
	    print "m1(x=%s)" %`x`
    print C
    x = C()
    print x
    x.m1(12)
    class D(C):
	def __getattr__(self, name):
	    if name[:2] == '__': raise AttributeError, name
	    return "getattr:%s" % name
    x = D()
    print x.foo
    print x._foo
##     print x.__foo
##     print x.__foo__


if __name__ == '__main__':
    _test()