如何通过工厂以扭曲协议发送数据

时间:2011-05-22 13:39:44

标签: python network-programming twisted

我正在编写一个实现自定义协议的客户端,并且有一个工厂。我的问题如下:我的客户端有双向通信,有时我想告诉它“发送此数据”。但我所拥有的只是工厂对象:

class MyFactory(ClientFactory):
    protocol = MyProtocol

    def __init__(self, recv_callback):
        self.recv_callback = recv_callback

    def send_message(self, msg):
        self.protocol.send_message(msg)

所以我创建了一个工厂并拥有一个工厂对象,我没有协议对象。如果调用上面的send_message,我会收到错误,因为self.protocol只是一个类,而不是一个对象。

我该怎么做?除了工厂之外,我还应该公开连接协议吗?

由于

1 个答案:

答案 0 :(得分:10)

您可以访问所需的所有对象。工厂负责创建协议实例,因此如果您希望将协议实例保留在工厂可以使用的位置,请覆盖buildProtocol并保存实例:

class MyFactory(ClientFactory):
    protocol = MyProtocol

    ...

    def buildProtocol(self, address):
        proto = ClientFactory.buildProtocol(self, address)
        self.connectedProtocol = proto
        return proto

然而,这种方法缺乏一个重要特征。在调用buildProtocol并设置connectedProtocol时,无法轻松判断。如果你试图天真地使用这个属性:

factory = MyFactory()
reactor.connectTCP(host, port, factory)
factory.connectedProtocol.send_message(...)

代码将失败并显示AttributeError,因为尚未实际建立连接。由于Twisted是事件驱动的,因此您需要确保通过响应表示已建立连接的事件来使用此代码。

您可以通过在构造协议时触发回调而不是仅设置属性来执行此操作。 Twisted实际上有一个帮助工厂,它已经做了类似的事情:

from twisted.internet.protocol import ClientCreator

cc = ClientCreator(reactor, MyProtocol)
whenConnected = cc.connectTCP(host, port)

# Or the equivalent with endpoints
#  from twisted.internet.endpoints import TCP4ClientEndpoint
#  from twisted.internet.protocol import ClientFactory
#  endpoint = TCP4ClientEndpoint(reactor, host, port)
#  factory = ClientFactory()
#  factory.protocol = MyProtocol
#  whenConnected = endpoint.connect(factory)

def cbConnected(connectedProtocol):
    connectedProtocol.send_message(...)

def ebConnectError(reason):
    # Connection attempt failed, perhaps retry
    ...

whenConnected.addCallbacks(cbConnected, ebConnectError)

您还可以在connectedProtocol回调中保存cbConnected的引用,以便稍后继续使用它。您也可以在cbConnected中启动任何其他想要使用连接协议的操作,这样他们就不会在实际可用之前尝试使用该连接。