1

我有一个简单的xmlrpc服务器定义为(server.py):

from SimpleXMLRPCServer import SimpleXMLRPCServer

def foo():
    return "foo"

server = SimpleXMLRPCServer(("localhost", 1025))
server.register_introspection_functions()
server.register_function(foo, 'test.foo')
server.serve_forever()

客户端 ( client.py) 实现如下:

import xmlrpclib

class XMLRPC(object):

    def __init__(self):
        self.xmlrpc = xmlrpclib.ServerProxy("http://localhost:1025/RPC2")

    def __getattr__(self, name):
        attr = getattr(self.xmlrpc, name)
        def wrapper(*args, **kwargs):
            result = attr(*args, **kwargs)
            return result
        return wrapper

xmlrpc = XMLRPC()
print xmlrpc.test.foo()

我想包装并执行xmlrpc类中对服务器进行的每个调用XMLRPC。但是上面的工作示例给出了一个错误

Traceback (most recent call last):
  File "client.py", line 20, in <module>
    xmlrpc.test.foo()
AttributeError: 'function' object has no attribute 'foo'

如何使这段代码工作?

附加信息和限制:

  • 我已经尝试过包装wrapperfunctools.wraps(attr)没有成功。由于字符串attr没有属性__name__,我得到一个不同的错误
  • 我无法更改server.py.
  • 上面的例子是完全有效的。
  • 替换return wrapperreturn attr不是解决方案 - 我需要xmlrpcwrapper.
  • 我需要一个没有 3rd 方库的简单解决方案,标准的 python 库是可以的。
4

1 回答 1

4

您需要返回一个可调用对象。

XML-RPC 代理返回一个可调用但也可以遍历的对象实例。因此,对于xmlrpc.test.foo(),您正在包装xmlrpc.test一个函数;该函数对象没有foo属性,因为函数通常没有这样的属性。

而是返回一个代理对象;钩子使它成为一个可调用的__call__对象,就像一个函数一样:

class WrapperProxy(object):
    def __init__(self, wrapped):
        self.wrapped = wrapped

    def __getattr__(self, name):
        attr = getattr(self.wrapped, name)
       return type(self)(attr)

    def __call__(self, *args, **kw):
        return self.wrapped(*args, **kw)

class XMLRPC(object):
    def __init__(self):
        self.xmlrpc = xmlrpclib.ServerProxy("http://localhost:1025/RPC2")

    def __getattr__(self, name):
        attr = getattr(self.xmlrpc, name)
        return WrapperProxy(attr)

或者,合并为一个对象:

class XMLRPCWrapperProxy(object):
    def __init__(self, wrapped=None):
        if wrapped is None: 
            wrapped = xmlrpclib.ServerProxy("http://localhost:1025/RPC2")
        self.wrapped = wrapped

    def __getattr__(self, name):
        attr = getattr(self.wrapped, name)
       return type(self)(attr)

    def __call__(self, *args, **kw):
        return self.wrapped(*args, **kw)
于 2013-06-25T12:33:32.337 回答