书:扭曲的网络编程要领,例9-1不起作用

时间:2013-10-04 01:25:00

标签: python authentication twisted telnet credentials

我要读Jessica McKellar和Abe Fetting所着的第二版扭曲网络编程要点。我被困在以下扭曲的版本示例9-1。我已经逐行检查了我的代码,逐字逐句。代码将运行,当我连接telnet客户端并输入'user pass'时,它不会登录。我只是得到未经授权的消息。它自己的脚本不会生成任何错误消息或异常。

由于

import sys
from zope.interface import implements, Interface

from twisted.cred import checkers, credentials, portal
from twisted.internet import protocol, reactor
from twisted.protocols import basic
from twisted.python.log import startLogging; startLogging(sys.stdout)

class IPortocolAvatar(Interface):
    def logout():
        """
        Clean up per-login resource allocated to this avatar.
        """

class EchoAvatar(object):
    implements(IPortocolAvatar)

    def logout(self):
        pass

class Echo(basic.LineReceiver):
    portal = None
    avatar = None
    logout = None

    def connectionLost(self, reason):
        if self.logout:
            self.logout()
            self.avatar = None
            self.logout = None

    def lineReceived(self, line):
        if not self.avatar:
            # print "line [32]: %s" % (line,)
            username, password = line.strip().split(" ")
            self.tryLogin(username, password)

        else:
            self.sendLine(line)

    def tryLogin(self, username, password):
        self.portal.login(credentials.UsernamePassword(username, password), None, PortocolAvatar).addCallbacks(self._cbLogin, self._ebLogin)

    def _cbLogin(self, (interface, avatar, logout)):
        self.avatar = avatar
        self.logout = logout
        self.sendLine("Login sucessful, please proceed.")

    def _ebLogin(self, failure):
        self.sendLine("Login denied, goodbye.")
        self.transport.loseConnection()

class EchoFactory(protocol.Factory):
    def __init__(self, portal):
        self.portal = portal

    def buildProtocol(self, addr):
        proto = Echo()
        proto.portal = self.portal
        return proto

class Realm(object):
    implements(portal.IRealm)

    def requestAvatar(self, avatarId, mind, *interfaces):
        if IPortocolAvatar in interfaces:
            avatar = EchoAvatar()
            return IPortocolAvatar, avatar, avatar.logout

        raise NotImplementedError(
            "This realm only supports the IPortocolAvatar interface.")

realm = Realm()
myPortal = portal.Portal(realm)
checker = checkers.InMemoryUsernamePasswordDatabaseDontUse()
checker.addUser("user", "pass")
myPortal.registerChecker(checker)

reactor.listenTCP(8000, EchoFactory(myPortal))
reactor.run()

2 个答案:

答案 0 :(得分:3)

您在tryLoginProtocolAvatar - > IProtocolAvatar)中输入了拼写错误:

def tryLogin(self, username, password):
    self.portal.login(credentials.UsernamePassword(username, password), None, IProtocolAvatar).addCallbacks(self._cbLogin, self._ebLogin)
    #                                                                         ^

<强>更新

还有另一个错字:IPortocolAvatar - &gt; IProtocolAvatar

运行服务器后,请尝试以下测试客户端代码:

import telnetlib
import time

t = telnetlib.Telnet('localhost', 8000)
t.write(b'user pass\r\n')
t.write(b'blah blah\r\n')
time.sleep(1)
print(t.read_eager())

答案 1 :(得分:0)

我也用python3运行了这个例子。 这是我改变的示例9-1的代码。 echo_cred.py&#39;来自“Twisted Network Programming Essentials,第2版&#39;

”一书
from zope.interface import Interface, implementer
from twisted.cred import checkers, credentials, portal, error
from twisted.internet import protocol, reactor, defer

from twisted.protocols import basic

class IProtocolAvatar(Interface):
    def logout():
        """
        Clean up per-login resources allocated to this avatar.
        """

@implementer(IProtocolAvatar)    
class EchoAvatar():

    def logout(self):
        pass

class Echo(basic.LineReceiver):
    portal = None
    avatar = None
    logout = None

    def connectionLost(self, reason):
        if self.logout:
            self.logout()
            self.avatar = None
            self.logout = None

    def lineReceived(self, line):
        print('line received')
        if not self.avatar:
            username, password = line.strip().split()
            # convert 2 text code
            username, password = username.decode(), password.decode()

            d = defer.maybeDeferred(self.tryLogin, username, password)
            d.addCallbacks(self._cbLogin, self._ebLogin)
        else:
            self.sendLine(line)

    def tryLogin(self, username, password):
        if self.portal is not None:
            return self.portal.login(
                credentials.UsernamePassword(username, password),
                None,
                IProtocolAvatar
            )
        raise error.UnauthorizedLogin()        

    def _cbLogin(self, ial):
        interface, avatar, logout = ial
        self.avatar = avatar
        self.logout = logout
        # convert 2 byte code
        self.sendLine("Login successful, please proceed.".encode())

    def _ebLogin(self, failure):
        self.sendLine("Login denied, goodbye.".encode())
        self.transport.loseConnection()

class EchoFactory(protocol.Factory):
    def __init__(self, portal):
        self.portal = portal

    def buildProtocol(self, addr):
        proto = Echo()
        proto.portal = self.portal
        return proto

@implementer(portal.IRealm)    
class Realm():

    def requestAvatar(self, avatarId, mind, *interfaces):
        if IProtocolAvatar in interfaces:
            avatar = EchoAvatar()
            return IProtocolAvatar, avatar, avatar.logout
        raise NotImplementedError("This realm only supports the IProtocolAvatar interface.")

if __name__ == '__main__':
    realm = Realm()
    myPortal = portal.Portal(realm)
    checker = checkers.InMemoryUsernamePasswordDatabaseDontUse()
    checker.addUser("user", "pass")
    myPortal.registerChecker(checker)
    reactor.listenTCP(8000, EchoFactory(myPortal))
    reactor.run()