使用Python的Twisted框架实现一个简单的服务器
预览
twisted是一个被设计的非常灵活框架以至于能够让你写出非常强大的服务器。这种灵活的代价是需要好通过好几个层次来实现你的服务器,本文档描述的是Protocol层,你将在这个层次中执行协议的分析和处理,如果你正在执行一个应用程序,那么你应该在读过toplevel的为twisted写插件一节中的怎样开始写twisted应用程序之后阅读本章。这个文档只是和TCP,SSL和Unix套接字服务器有关,同时也将有另一份文档专门讲解UDP。
你的协议处理类通常是twisted.internet.protocol.Protocol的子类。许多协议处理继承于该类或者比该类更加方便的该类的子类。一个protocol类的实例可能反复连接,也可能在连接关闭之后销毁。这就意味着这些持续不断的配置信息不是保存在Protocol中。
这些持久性的配置被保存在工厂(Factory)类中,这些工厂类通常继承至twisted.internet.protocol.Factory,默认的工厂类仅仅是实例化每个Protocol,然后设置他们的factory属性为这个默认的工厂实例本身。这就让每个Protocol都被存储,然后可能修改,于是这样就形成了Protocol的持久性。
通常为多个端口或网络地址提供相同的服务是非常有用的。这就是为什么Factory不监听连接,并且实际上它不知道关于网络的任何事情。看twisted.internet.interfaces.IReactorTCP.listenTCP,另一个IReactor*.listen*获得更多的信息。
本文档将要讲解各个步骤。
Protocol
如上所述,这里将通过更多代码的辅助类和函数来了解它。一个twistedprotocl通过异步方式处理数据。这就意味着protocol从不等待任何事件。相反的是在事件通过网络到达的时候作出响应。
fromtwisted.internet.protocolimportProtocol classEcho(Protocol): defdataReceived(self,data): self.transport.writed(data)
这是个非常简单的协议处理,仅仅是在获得数据的事件中简单的将接收到的数据发送回去,并没有对所有的事件进行响应。这里有一个Protocol响应其他事件的例子如下:
fromtwisted.internet.protocolimportProtocol classQOTD(Protocol): defconnectionMade(self): self.transport.write("Anappleadaykeepsthedoctoraway/r/n") self.transport.loseConnection()
本Protocl在一个已知的引用刚开始连接上来的时候作出响应,发送了一条消息,然后终止了连接connectionMade事件通常是在由于连接对象建立初始连接时触发,就像上面的QOTD类实际上是RFC865号文档的一个协议基类connectionLost事件将在断开连接的时候触发。实例:
<spanstyle="font-family:Monospaced;color:#0000a0;"><strong>PythonCode:</strong></span><tablestyle="width:100%;height:20px;"align="center"bgcolor="#e3dfe3"border="1"bordercolor="#9da7ac"cellpadding="0"cellspacing="0"> <tbody><tr><td> <divclass="textBackGround"style="font-family:CourierNew;font-size:9pt;"><pre><spanstyle="color:blue;">from</span>twisted.internet.protocol<spanstyle="color:blue;">import</span>Protocol <spanstyle="color:blue;">class</span>Echo(Protocol): <spanstyle="color:blue;">def</span>connectionMade(self): self.factory.numProtocols=self.factory.numProtocols+1 <spanstyle="color:blue;">if</span>self.factory.numProtocols>100: self.transport.write(<spanstyle="color:#ff44a2;">"Toomanyconnections,<spanstyle="color:blue;">try</span>later"</span>) self.transport.loseConnection() <spanstyle="color:blue;">def</span>connectionLost(self,reason): self.factory.numProtocols=self.factory.numProtocols-1 <spanstyle="color:blue;">def</span>dataReceived(self,data): self.transport.write(data)</pre> </div> </td> </tr> </tbody> </table>
本实例中,connectionMade和connectionLost相互协作工作以保持factory内部的活动连接数量最多为100。每当有用户协议连接近来的时候,就先检测factory内部的活动连接数,如果数量超过100,就发送连接数太多等下试的消息,然后断开连接而connectionLost则在断开一个协议的时候触发,减去factory内部的协议数量。
UsingtheProtocol
在本节,我将要讲解怎样简单的去测试你的protocol。(想知道如何写出一个好的twisted的服务器,请看<ahref="http://fantix.org/twisted-doc-zh/nightly/online/howto/plugin.html">WritingPlug-Ins<br> forTwisted</a>),这里有一个代码将运行我们上面谈论的QOTD服务器:
<!-- .textBackGround{background-color:#F0F5FD;} --> <spanstyle="font-family:Monospaced;color:#0000a0;"><strong>PythonCode:</strong></span><tablestyle="width:100%;height:20px;"align="center"bgcolor="#e3dfe3"border="1"bordercolor="#9da7ac"cellpadding="0"cellspacing="0"> <tbody><tr><td> <divclass="textBackGround"style="font-family:CourierNew;font-size:9pt;"><pre><spanstyle="color:blue;">from</span>twisted.internet.protocol<spanstyle="color:blue;">import</span>Protocol,Factory <spanstyle="color:blue;">from</span>twisted.internet<spanstyle="color:blue;">import</span>reactor <spanstyle="color:blue;">class</span>QOTD(Protocol): <spanstyle="color:blue;">def</span>connectionMade(self): self.transport.write(<spanstyle="color:#ff44a2;">"Anappleadaykeepsthedoctoraway/r/n"</span>) self.transport.loseConnection() <spanstyle="color:green;">#Nextlinesaremagic:</span> factory=Factory() factory.protocol=QOTD <spanstyle="color:green;">#8007<spanstyle="color:blue;">is</span>theportyouwanttorununder.Choosesomething>1024</span> reactor.listenTCP(8007,factory) reactor.run()</pre> </div> </td> </tr> </tbody> </table>
不必担心最后面的6条代码,稍后你将会在本文档中了解到他们。<br>
HelperProtocols
大部分protocols依赖于同类别的更低层次的超级类。最受欢迎的互联网协议是基于行,行通常是由CR_LF(回车换行组成)
然而,也有相当一部分协议是混合的,他们具有线性的基本节点,也有原始数据节点,比如HTTP/1.1。
在这样的情况下,我们可以使用LineReceiver,本协议类有两个不同的事件处理方法,lineReceived和rawDataReceived
默认情况下,只有lineReceived会被调用,每次读取一行,然而如果setRawMode被调用,protocol将调用rawDataReceived
来处理直到setLineMode被调用。下面有一个简单的例子说明如何使用lineReceiver:
PythonCode:
fromtwisted.protocols.basicimportLineReceiver classAnswer(LineReceiver): answers={'Howareyou?':'Fine',None:"Idon'tknowwhatyoumean"} deflineReceived(self,line): ifself.answers.has_key(line): self.sendLine(self.answers[line]) else: self.sendLine(self.answers[None])
注意:界定符不是命令行的一部分
其他也有一些不流行的协议依然存在,比如netstringbased和aprefixed-message-length
StateMachines
许多twistedprotocolhandlers需要编写一个状态机来记录他们当前的状态,这里有几点编写状态机的建议:
1、不要编写大状态机,宁愿去实现一个抽象的状态机类
2、使用python的动态性质去创建没有限制的状态机,比如SMTP客户端
3、不要混合特定应用程序代码和协议处理代码,当协议处理器已经提出一个特别的具体要求,保持它作为一个方法调用。
Factories(工厂类)
如前面所说,通常twisted.internet.protocol.Factory不必子类化就可以开始工作。然而有时候protocol需要具体的
特殊的工厂配置信息或其他需求,在这样的情况下,就需要进行子类化了。
对于Factory来说,他只是简单的实例化特殊的protocol协议类,实例化Factory,并且设置protocol属性:
PythonCode:
fromtwisted.internet.protocolimportFactory fromtwisted.protocols.wireimportEcho myFactory=Factory() myFactory.protocol=Echo
如果需要简单的去构造一个有具体特殊信息的工厂类,那么一个factory函数是非常有用的:
PythonCode:
classQOTD(Protocol): defconnectionMade(self): self.transport.write(self.factory.quote+'/r/n') self.transport.loseConnection() defmakeQOTDFactory(quote=None): factory=Factory() factory.protocol=QOTD factory.quote=quoteor'Anappleadaykeepsthedoctoraway' returnfactory
一个Factory有两个方法以执行特定于应用程序的建立和拆除(由于一个Factory通常存在,所以常规下一般不在__init__或者
__del__中给他们分配与回收,有可能太早或太晚)。
下面是一个Factory的例子,本例将允许Protocol写一个日志文件:
PythonCode:
fromtwisted.internet.protocolimportFactory fromtwisted.protocols.basicimportLineReceiver classLoggingProtocol(LineReceiver): deflineReceived(self,line): self.factory.fp.write(line+'/n') classLogfileFactory(Factory): protocol=LoggingProtocol def__init__(self,fileName): self.file=fileName defstartFactory(self): self.fp=open(self.file,'a') defstopFactory(self): self.fp.close()
PuttingitAllTogether(综合)
现在你已经了解了Factory并且想要执行QOTD作为一个可配置的quote服务器是吗?没有问题这里就有一个代码:
PythonCode:
fromtwisted.internet.protocolimportFactory,Protocol fromtwisted.internetimportreactor classQOTD(Protocol): defconnectionMade(self): self.transport.write(self.factory.quote+'/r/n') self.transport.loseConnection() classQOTDFactory(Factory): protocol=QOTD def__init__(self,quote=None): self.quote=quoteor'Anappleadaykeepsthedoctoraway' reactor.listenTCP(8007,QOTDFactory("configurablequote")) reactor.run()
就是最后两句代码,还需要去理解。
listenTCP是一个将Factory连接到网络的方法,他使用了reactor的接口,让许多不同的循环处理网络代码,而不需要修改的
最终用户代码,就像这样。如前面所说,如果你想要写一个好的twisted服务器,而不是仅仅的20行,那么你需要使用theApplicationobject.