commit df8d9b46a3d4e6d258afb49e45b68c2a75abbe13 Author: Arturo Filastò art@fuffa.org Date: Mon Feb 3 16:29:14 2014 +0100
Write test that reproduces HTTP test stalling condition.
The condition would happen when a HTTP server would keep the connection open without sending any data. The problem lied in the fact that we were not properly cancelling tests that should have been cancelled.
This was happening because of 2 reasons: 1) We should not be checking to see if self._running.called is True since it is possible that such deferred has been called, but the it has not yet made it's way down the callback chain to the ooni registered callbacks. 2) We must call the cancel() method after we have called the failed() method. Failing to do so will lead to the newly scheduled timer to be cancelled instead of the previous one. --- ooni/tasks.py | 5 ++-- ooni/tests/test_nettest.py | 71 ++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 73 insertions(+), 3 deletions(-)
diff --git a/ooni/tasks.py b/ooni/tasks.py index 3e378e9..6e8c507 100644 --- a/ooni/tasks.py +++ b/ooni/tasks.py @@ -69,12 +69,11 @@ class TaskWithTimeout(BaseTask):
def _timedOut(self): """Internal method for handling timeout failure""" - if self._running and not self._running.called: - self._running.cancel() + if self._running: self._failed(TaskTimedOut) + self._running.cancel()
def _cancelTimer(self): - #import pdb; pdb.set_trace() if self._timer.active(): self._timer.cancel()
diff --git a/ooni/tests/test_nettest.py b/ooni/tests/test_nettest.py index d965735..718fca7 100644 --- a/ooni/tests/test_nettest.py +++ b/ooni/tests/test_nettest.py @@ -6,6 +6,7 @@ from twisted.trial import unittest from twisted.internet import defer, reactor from twisted.python.usage import UsageError
+from ooni.settings import config from ooni.errors import MissingRequiredOption, InvalidOption, FailureToLoadNetTest from ooni.nettest import NetTest, NetTestLoader from ooni.tasks import BaseTask @@ -83,6 +84,28 @@ class DummyTestCase(NetTestCase): requiredOptions = ['foo', 'bar'] """
+http_net_test = """ +from twisted.internet import defer +from twisted.python import usage, failure + +from ooni.utils import log +from ooni.utils.net import userAgents +from ooni.templates import httpt +from ooni.errors import failureToString, handleAllFailures + +class UsageOptions(usage.Options): + optParameters = [ + ['url', 'u', None, 'Specify a single URL to test.'], + ['factor', 'f', 0.8, 'What factor should be used for triggering censorship (0.8 == 80%)'] + ] + +class HTTPBasedTest(httpt.HTTPTest): + usageOptions = UsageOptions + def test_get(self): + return self.doRequest(self.localOptions['url'], method="GET", + use_tor=False) +""" + dummyInputs = range(1) dummyArgs = ('--spam', 'notham') dummyOptions = {'spam':'notham'} @@ -243,3 +266,51 @@ class TestNetTest(unittest.TestCase):
for test_class, method in ntl.testCases: self.assertTrue(test_class.requiresRoot) + +class TestNettestTimeout(unittest.TestCase): + @defer.inlineCallbacks + def setUp(self): + from twisted.internet.protocol import Protocol, Factory + from twisted.internet.endpoints import TCP4ServerEndpoint + + class DummyProtocol(Protocol): + def dataReceived(self, data): + print data + + class DummyFactory(Factory): + def __init__(self): + self.protocols = [] + + def buildProtocol(self, addr): + proto = DummyProtocol() + self.protocols.append(proto) + return proto + + def stopFactory(self): + for proto in self.protocols: + proto.transport.loseConnection() + + self.factory = DummyFactory() + endpoint = TCP4ServerEndpoint(reactor, 8007) + self.port = yield endpoint.listen(self.factory) + + config.advanced.measurement_timeout = 2 + + def tearDown(self): + self.factory.stopFactory() + self.port.stopListening() + + def test_nettest_timeout(self): + ntl = NetTestLoader(('-u', 'http://localhost:8007/')) + ntl.loadNetTestString(http_net_test) + + ntl.checkOptions() + director = Director() + + d = director.startNetTest(ntl, [MockReporter()]) + + @d.addCallback + def complete(result): + assert director.failedMeasurements == 1 + + return d
tor-commits@lists.torproject.org