Source code for ldaptor.testutil

"""Utilities for writing Twistedy unit tests and debugging."""

from twisted.internet import defer
from twisted.python import failure
from twisted.trial import unittest
from twisted.test import proto_helpers
from ldaptor import config

[docs]def mustRaise(dummy): raise unittest.FailTest('Should have raised an exception.')
[docs]def calltrace(): """Print out all function calls. For debug use only.""" def printfuncnames(frame, event, arg): print "|%s: %s:%d:%s" % (event, frame.f_code.co_filename, frame.f_code.co_firstlineno, frame.f_code.co_name) import sys sys.setprofile(printfuncnames)
[docs]class FakeTransport: def __init__(self, proto): self.proto = proto
[docs] def loseConnection(self): self.proto.connectionLost()
[docs]class LDAPClientTestDriver: """ A test driver that looks somewhat like a real LDAPClient. Pass in a list of lists of LDAPProtocolResponses. For each sent LDAP message, the first item of said list is iterated through, and all the items are sent as responses to the callback. The sent LDAP messages are stored in self.sent, so you can assert that the sent messages are what they are supposed to be. It is also possible to include a Failure instance instead of a list of LDAPProtocolResponses which will cause the errback to be called with the failure. """ def __init__(self, *responses): self.sent=[] self.responses=list(responses) self.connected = None self.transport = FakeTransport(self)
[docs] def send(self, op): self.sent.append(op) l = self._response() assert len(l) == 1, \ "got %d responses for a .send()" % len(l) r = l[0] if isinstance(r, failure.Failure): return defer.fail(r) else: return defer.succeed(r)
[docs] def send_multiResponse(self, op, handler, *args, **kwargs): d = defer.Deferred() self.sent.append(op) responses = self._response() while responses: r = responses.pop(0) if isinstance(r, failure.Failure): d.errback(r) break ret = handler(r, *args, **kwargs) if responses: assert not ret, \ "got %d responses still to give, but handler wants none (got %r)." % (len(responses), ret) else: assert ret, \ "no more responses to give, but handler still wants more (got %r)." % ret return d
[docs] def send_noResponse(self, op): responses = self.responses.pop(0) assert not responses self.sent.append(op)
def _response(self): assert self.responses, 'Ran out of responses' responses = self.responses.pop(0) return responses
[docs] def assertNothingSent(self): # just a bit more explicit self.assertSent()
[docs] def assertSent(self, *shouldBeSent): shouldBeSent = list(shouldBeSent) assert self.sent == shouldBeSent, \ '%s expected to send %r but sent %r' % ( self.__class__.__name__, shouldBeSent, self.sent) sentStr = ''.join([str(x) for x in self.sent]) shouldBeSentStr = ''.join([str(x) for x in shouldBeSent]) assert sentStr == shouldBeSentStr, \ '%s expected to send data %r but sent %r' % ( self.__class__.__name__, shouldBeSentStr, sentStr)
[docs] def connectionMade(self): """TCP connection has opened""" self.connected = 1
[docs] def connectionLost(self, reason=None): """Called when TCP connection has been lost""" assert not self.responses, \ "connectionLost called even when have responses left: %r" % self.responses self.connected = 0
[docs] def unbind(self): assert self.connected r='fake-unbind-by-LDAPClientTestDriver' self.send_noResponse(r) self.transport.loseConnection()
[docs]def createServer(proto, *responses, **kw): def createClient(factory): factory.doStart() #TODO factory.startedConnecting(c) proto = factory.buildProtocol(addr=None) proto.connectionMade() overrides = kw.setdefault('serviceLocationOverrides', {}) overrides.setdefault('', createClient) conf = config.LDAPConfig(**kw) server = proto(conf) server.protocol = lambda : LDAPClientTestDriver(*responses) server.transport = proto_helpers.StringTransport() server.connectionMade() return server