[tor-commits] [ooni-probe/master] Import basic stuff from Twisted trial

isis at torproject.org isis at torproject.org
Thu Oct 4 14:41:15 UTC 2012


commit c4ad2411046deb08f17902320cb8dd21cc25f367
Author: Arturo Filastò <arturo at filasto.net>
Date:   Tue Sep 11 10:38:35 2012 +0000

    Import basic stuff from Twisted trial
---
 .gitignore                  |    1 +
 README.md                   |   10 +
 bin/ooniprobe               |   16 ++-
 ooni/oonicli.py             |  397 +++++++++++++++++++++++++++++++++++++++++++
 ooni/oonitests/bridget.py   |  373 ----------------------------------------
 ooni/plugins/new_bridget.py |    2 +-
 ooni/protocols/http.py      |   13 +-
 ooni/reporter.py            |   13 ++
 ooni/runner.py              |    7 +
 9 files changed, 448 insertions(+), 384 deletions(-)

diff --git a/.gitignore b/.gitignore
index 7f270bb..6b1b9ce 100644
--- a/.gitignore
+++ b/.gitignore
@@ -8,3 +8,4 @@ proxy-lists/italy-http-ips.txt
 private/*
 /ooni/plugins/dropin.cache
 oonib/oonibackend.conf
+ooni/assets/*
diff --git a/README.md b/README.md
index b39cb2b..123d837 100644
--- a/README.md
+++ b/README.md
@@ -28,6 +28,16 @@ To list the help for a specific test:
 
     python ooniprobe.py httpt --help
 
+## Virtualenv way (Recommended)
+
+    virtualenv2 ENV/
+    source ENV/bin/activate
+    pip install twisted Scapy
+
+To install the most up to date scapy version (requires mercurial):
+
+    pip install hg+http://hg.secdev.org/scapy
+
 
 # More details
 
diff --git a/bin/ooniprobe b/bin/ooniprobe
index 5c87831..1f0c26d 100755
--- a/bin/ooniprobe
+++ b/bin/ooniprobe
@@ -1,5 +1,11 @@
-#!/bin/sh
-ROOT="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )"
-echo $ROOT
-export PYTHONPATH=$PYTHONPATH:$ROOT
-python $ROOT/ooni/ooniprobe.py $1
+#!/usr/bin/python2
+# startup script based on twisted trial
+# See http://twistedmatrix.com/
+import os, sys
+
+sys.path[:] = map(os.path.abspath, sys.path)
+
+sys.path.insert(0, os.path.abspath(os.getcwd()))
+
+from ooni.oonicli import run
+run()
diff --git a/ooni/oonicli.py b/ooni/oonicli.py
new file mode 100644
index 0000000..8ace160
--- /dev/null
+++ b/ooni/oonicli.py
@@ -0,0 +1,397 @@
+#!/usr/bin/env python
+# -*- coding: UTF-8
+#
+#    oonicli
+#    *********
+#
+#    oonicli is the next generation ooniprober. It based off of twisted's trial
+#    unit testing framework.
+#
+#    :copyright: (c) 2012 by Arturo Filastò
+#    :license: see LICENSE for more details.
+#
+#    original copyright (c) by Twisted Matrix Laboratories.
+
+
+import sys, os, random, gc, time, warnings
+
+from twisted.internet import defer
+from twisted.application import app
+from twisted.python import usage, reflect, failure
+from twisted.python.filepath import FilePath
+from twisted import plugin
+from twisted.python.util import spewer
+from twisted.python.compat import set
+from twisted.trial import runner, itrial, reporter
+
+
+# Yea, this is stupid.  Leave it for for command-line compatibility for a
+# while, though.
+TBFORMAT_MAP = {
+    'plain': 'default',
+    'default': 'default',
+    'emacs': 'brief',
+    'brief': 'brief',
+    'cgitb': 'verbose',
+    'verbose': 'verbose'
+    }
+
+
+def _parseLocalVariables(line):
+    """
+    Accepts a single line in Emacs local variable declaration format and
+    returns a dict of all the variables {name: value}.
+    Raises ValueError if 'line' is in the wrong format.
+
+    See http://www.gnu.org/software/emacs/manual/html_node/File-Variables.html
+    """
+    paren = '-*-'
+    start = line.find(paren) + len(paren)
+    end = line.rfind(paren)
+    if start == -1 or end == -1:
+        raise ValueError("%r not a valid local variable declaration" % (line,))
+    items = line[start:end].split(';')
+    localVars = {}
+    for item in items:
+        if len(item.strip()) == 0:
+            continue
+        split = item.split(':')
+        if len(split) != 2:
+            raise ValueError("%r contains invalid declaration %r"
+                             % (line, item))
+        localVars[split[0].strip()] = split[1].strip()
+    return localVars
+
+
+def loadLocalVariables(filename):
+    """
+    Accepts a filename and attempts to load the Emacs variable declarations
+    from that file, simulating what Emacs does.
+
+    See http://www.gnu.org/software/emacs/manual/html_node/File-Variables.html
+    """
+    f = file(filename, "r")
+    lines = [f.readline(), f.readline()]
+    f.close()
+    for line in lines:
+        try:
+            return _parseLocalVariables(line)
+        except ValueError:
+            pass
+    return {}
+
+
+def getTestModules(filename):
+    testCaseVar = loadLocalVariables(filename).get('test-case-name', None)
+    if testCaseVar is None:
+        return []
+    return testCaseVar.split(',')
+
+
+def isTestFile(filename):
+    """
+    Returns true if 'filename' looks like a file containing unit tests.
+    False otherwise.  Doesn't care whether filename exists.
+    """
+    basename = os.path.basename(filename)
+    return (basename.startswith('test_')
+            and os.path.splitext(basename)[1] == ('.py'))
+
+
+def _reporterAction():
+    return usage.CompleteList([p.longOpt for p in
+                               plugin.getPlugins(itrial.IReporter)])
+
+class Options(usage.Options):
+
+    optParameters = [
+        ['parallelism', 'n', 10, "Specify the number of parallel tests to run"],
+        ['output', 'o', 'report.log', "Specify output report file"],
+        ['log', 'l', 'oonicli.log', "Specify output log file"]
+    ]
+
+    def opt_version(self):
+        """
+        Display OONI version and exit.
+        """
+        print "OONI version:", __version__
+        sys.exit(0)
+
+
+class Options(usage.Options, app.ReactorSelectionMixin):
+    synopsis = """%s [options] [[file|package|module|TestCase|testmethod]...]
+    """ % (os.path.basename(sys.argv[0]),)
+
+    longdesc = ("ooniprobe loads and executes a suite or a set of suites of"
+                "network tests. These are loaded from modules, packages and"
+                "files listed on the command line")
+
+    optFlags = [["help", "h"],
+                ["rterrors", "e", "realtime errors, print out tracebacks as "
+                 "soon as they occur"],
+                ["debug", "b", "Run tests in the Python debugger. Will load "
+                 "'.pdbrc' from current directory if it exists."],
+                ["debug-stacktraces", "B", "Report Deferred creation and "
+                 "callback stack traces"],
+                ["nopm", None, "don't automatically jump into debugger for "
+                 "postmorteming of exceptions"],
+                ["dry-run", 'n', "do everything but run the tests"],
+                ["force-gc", None, "Have Trial run gc.collect() before and "
+                 "after each test case."],
+                ["profile", None, "Run tests under the Python profiler"],
+                ["unclean-warnings", None,
+                 "Turn dirty reactor errors into warnings"],
+                ["until-failure", "u", "Repeat test until it fails"],
+                ["no-recurse", "N", "Don't recurse into packages"],
+                ['help-reporters', None,
+                 "Help on available output plugins (reporters)"]
+                ]
+
+    optParameters = [
+        ["logfile", "l", "test.log", "log file name"],
+        ["random", "z", None,
+         "Run tests in random order using the specified seed"],
+        ['temp-directory', None, '_trial_temp',
+         'Path to use as working directory for tests.'],
+        ['reporter', None, 'verbose',
+         'The reporter to use for this test run.  See --help-reporters for '
+         'more info.']]
+
+    compData = usage.Completions(
+        optActions={"tbformat": usage.CompleteList(["plain", "emacs", "cgitb"]),
+                    "reporter": _reporterAction,
+                    "logfile": usage.CompleteFiles(descr="log file name"),
+                    "random": usage.Completer(descr="random seed")},
+        extraActions=[usage.CompleteFiles(
+                "*.py", descr="file | module | package | TestCase | testMethod",
+                repeat=True)],
+        )
+
+    fallbackReporter = reporter.TreeReporter
+    tracer = None
+
+    def __init__(self):
+        self['tests'] = set()
+        usage.Options.__init__(self)
+
+
+    def coverdir(self):
+        """
+        Return a L{FilePath} representing the directory into which coverage
+        results should be written.
+        """
+        coverdir = 'coverage'
+        result = FilePath(self['temp-directory']).child(coverdir)
+        print "Setting coverage directory to %s." % (result.path,)
+        return result
+
+
+    def opt_coverage(self):
+        """
+        Generate coverage information in the I{coverage} file in the
+        directory specified by the I{trial-temp} option.
+        """
+        import trace
+        self.tracer = trace.Trace(count=1, trace=0)
+        sys.settrace(self.tracer.globaltrace)
+
+
+    def opt_testmodule(self, filename):
+        """
+        Filename to grep for test cases (-*- test-case-name)
+        """
+        # If the filename passed to this parameter looks like a test module
+        # we just add that to the test suite.
+        #
+        # If not, we inspect it for an Emacs buffer local variable called
+        # 'test-case-name'.  If that variable is declared, we try to add its
+        # value to the test suite as a module.
+        #
+        # This parameter allows automated processes (like Buildbot) to pass
+        # a list of files to Trial with the general expectation of "these files,
+        # whatever they are, will get tested"
+        if not os.path.isfile(filename):
+            sys.stderr.write("File %r doesn't exist\n" % (filename,))
+            return
+        filename = os.path.abspath(filename)
+        if isTestFile(filename):
+            self['tests'].add(filename)
+        else:
+            self['tests'].update(getTestModules(filename))
+
+
+    def opt_spew(self):
+        """
+        Print an insanely verbose log of everything that happens.  Useful
+        when debugging freezes or locks in complex code.
+        """
+        sys.settrace(spewer)
+
+
+    def opt_help_reporters(self):
+        synopsis = ("OONI's output can be customized using plugins called "
+                    "Reporters. You can\nselect any of the following "
+                    "reporters using --reporter=<foo>\n")
+        print synopsis
+        for p in plugin.getPlugins(itrial.IReporter):
+            print '   ', p.longOpt, '\t', p.description
+        print
+        sys.exit(0)
+
+
+    def opt_disablegc(self):
+        """
+        Disable the garbage collector
+        """
+        gc.disable()
+
+
+    def opt_tbformat(self, opt):
+        """
+        Specify the format to display tracebacks with. Valid formats are
+        'plain', 'emacs', and 'cgitb' which uses the nicely verbose stdlib
+        cgitb.text function
+        """
+        try:
+            self['tbformat'] = TBFORMAT_MAP[opt]
+        except KeyError:
+            raise usage.UsageError(
+                "tbformat must be 'plain', 'emacs', or 'cgitb'.")
+
+
+    def opt_recursionlimit(self, arg):
+        """
+        see sys.setrecursionlimit()
+        """
+        try:
+            sys.setrecursionlimit(int(arg))
+        except (TypeError, ValueError):
+            raise usage.UsageError(
+                "argument to recursionlimit must be an integer")
+
+
+    def opt_random(self, option):
+        try:
+            self['random'] = long(option)
+        except ValueError:
+            raise usage.UsageError(
+                "Argument to --random must be a positive integer")
+        else:
+            if self['random'] < 0:
+                raise usage.UsageError(
+                    "Argument to --random must be a positive integer")
+            elif self['random'] == 0:
+                self['random'] = long(time.time() * 100)
+
+
+    def opt_without_module(self, option):
+        """
+        Fake the lack of the specified modules, separated with commas.
+        """
+        for module in option.split(","):
+            if module in sys.modules:
+                warnings.warn("Module '%s' already imported, "
+                              "disabling anyway." % (module,),
+                              category=RuntimeWarning)
+            sys.modules[module] = None
+
+
+    def parseArgs(self, *args):
+        self['tests'].update(args)
+
+
+    def _loadReporterByName(self, name):
+        for p in plugin.getPlugins(itrial.IReporter):
+            qual = "%s.%s" % (p.module, p.klass)
+            if p.longOpt == name:
+                return reflect.namedAny(qual)
+        raise usage.UsageError("Only pass names of Reporter plugins to "
+                               "--reporter. See --help-reporters for "
+                               "more info.")
+
+
+    def postOptions(self):
+        # Only load reporters now, as opposed to any earlier, to avoid letting
+        # application-defined plugins muck up reactor selecting by importing
+        # t.i.reactor and causing the default to be installed.
+        self['reporter'] = self._loadReporterByName(self['reporter'])
+
+        if 'tbformat' not in self:
+            self['tbformat'] = 'default'
+        if self['nopm']:
+            if not self['debug']:
+                raise usage.UsageError("you must specify --debug when using "
+                                       "--nopm ")
+            failure.DO_POST_MORTEM = False
+
+
+
+def _initialDebugSetup(config):
+    # do this part of debug setup first for easy debugging of import failures
+    if config['debug']:
+        failure.startDebugMode()
+    if config['debug'] or config['debug-stacktraces']:
+        defer.setDebugging(True)
+
+
+
+def _getSuite(config):
+    loader = _getLoader(config)
+    recurse = not config['no-recurse']
+    return loader.loadByNames(config['tests'], recurse)
+
+
+
+def _getLoader(config):
+    loader = runner.TestLoader()
+    if config['random']:
+        randomer = random.Random()
+        randomer.seed(config['random'])
+        loader.sorter = lambda x : randomer.random()
+        print 'Running tests shuffled with seed %d\n' % config['random']
+    if not config['until-failure']:
+        loader.suiteFactory = runner.DestructiveTestSuite
+    return loader
+
+
+
+def _makeRunner(config):
+    mode = None
+    if config['debug']:
+        mode = runner.TrialRunner.DEBUG
+    if config['dry-run']:
+        mode = runner.TrialRunner.DRY_RUN
+    return runner.TrialRunner(config['reporter'],
+                              mode=mode,
+                              profile=config['profile'],
+                              logfile=config['logfile'],
+                              tracebackFormat=config['tbformat'],
+                              realTimeErrors=config['rterrors'],
+                              uncleanWarnings=config['unclean-warnings'],
+                              workingDirectory=config['temp-directory'],
+                              forceGarbageCollection=config['force-gc'])
+
+
+
+def run():
+    if len(sys.argv) == 1:
+        sys.argv.append("--help")
+    config = Options()
+    try:
+        config.parseOptions()
+    except usage.error, ue:
+        raise SystemExit, "%s: %s" % (sys.argv[0], ue)
+    _initialDebugSetup(config)
+    trialRunner = _makeRunner(config)
+    suite = _getSuite(config)
+    if config['until-failure']:
+        test_result = trialRunner.runUntilFailure(suite)
+    else:
+        test_result = trialRunner.run(suite)
+    if config.tracer:
+        sys.settrace(None)
+        results = config.tracer.results()
+        results.write_results(show_missing=1, summary=False,
+                              coverdir=config.coverdir().path)
+    sys.exit(not test_result.wasSuccessful())
+
diff --git a/ooni/oonitests/bridget.py b/ooni/oonitests/bridget.py
deleted file mode 100644
index a613f61..0000000
--- a/ooni/oonitests/bridget.py
+++ /dev/null
@@ -1,373 +0,0 @@
-# -*- coding: UTF-8
-"""
-    bridgeT
-    *******
-
-    an OONI test (we call them Plugoos :P) aimed
-    at detecting if a set of Tor bridges are working or not.
-
-    :copyright: (c) 2012 by Arturo Filastò
-    :license: BSD, see LICENSE for more details.
-"""
-import os
-import sys
-import errno
-import time
-import random
-import re
-import glob
-import socks
-import socket
-from shutil import rmtree
-from subprocess import Popen, PIPE
-from datetime import datetime
-
-import shutil
-import gevent
-from gevent import socket
-import fcntl
-from plugoo.assets import Asset
-from plugoo.tests import Test
-import urllib2
-import httplib
-import json
-
-try:
-    from TorCtl import TorCtl
-except:
-    print "Error TorCtl not installed!"
-
-__plugoo__ = "BridgeT"
-__desc__ = "BridgeT, for testing Tor Bridge reachability"
-ONIONOO_URL="http://85.214.195.203/summary/search/"
-
-class SocksiPyConnection(httplib.HTTPConnection):
-    def __init__(self, proxytype, proxyaddr, proxyport = None, rdns = True,
-                 username = None, password = None, *args, **kwargs):
-        self.proxyargs = (proxytype, proxyaddr, proxyport, rdns, username, password)
-        httplib.HTTPConnection.__init__(self, *args, **kwargs)
-
-    def connect(self):
-        self.sock = socks.socksocket()
-        self.sock.setproxy(*self.proxyargs)
-        if isinstance(self.timeout, float):
-            self.sock.settimeout(self.timeout)
-        self.sock.connect((self.host, self.port))
-
-class SocksiPyHandler(urllib2.HTTPHandler):
-    def __init__(self, *args, **kwargs):
-        self.args = args
-        self.kw = kwargs
-        urllib2.HTTPHandler.__init__(self)
-
-    def http_open(self, req):
-        def build(host, port=None, strict=None, timeout=0):
-            conn = SocksiPyConnection(*self.args, host=host, port=port,
-                                      strict=strict, timeout=timeout, **self.kw)
-            return conn
-        return self.do_open(build, req)
-
-class BridgeTAsset(Asset):
-    def __init__(self, file=None):
-        self = Asset.__init__(self, file)
-
-class BridgeT(Test):
-    # This is the timeout value after which
-    # we will give up
-    timeout = 20
-    # These are the modules that should be torified
-    modules = [urllib2]
-
-    def tor_greater_than(self, version):
-        """
-        Checks if the currently installed version of Tor is greater
-        than the required version.
-
-        :version The version of Tor to check against for greater than or equal
-        """
-        fullstring = os.popen("tor --version").read().split('\n')[-2]
-        v_array = fullstring.split(' ')[2].split('-')
-        minor = v_array[1:]
-        v_array = v_array[0].split('.')
-        minor_p = version.split('-')[1:]
-        v_array_p = version.split('-')[0].split('.')
-
-        for i, x in enumerate(v_array):
-            try:
-                if i > len(v_array_p):
-                    break
-
-                if int(x) > int(v_array_p[i]):
-                    self.logger.debug("The Tor version is greater than %s" % version)
-                    return True
-                elif int(x) == int(v_array_p[i]):
-                    self.logger.debug("The Tor version is greater than %s" % version)
-                    continue
-                else:
-                    self.logger.debug("You run an outdated version of Tor: %s (< %s)" % (fullstring, version))
-                    return False
-            except:
-                self.logger.error("Error in parsing your Tor version string: %s" % fullstring)
-                return False
-
-        self.logger.debug("The Tor version is equal to %s" % version)
-        return True
-        # XXX currently don't consider the minor parts of the version
-        # (alpha, dev, beta, etc.)
-
-    def free_port(self, port):
-        s = socket.socket()
-        try:
-            s.bind(('127.0.0.1', port))
-            s.close()
-            return True
-        except:
-            self.logger.warn("The randomly chosen port was already taken!")
-            s.close()
-            return False
-
-    def writetorrc(self, bridge):
-        """
-        Write the torrc file for the tor process to be used
-        to test the bridge.
-
-        :bridge the bridge to be tested
-        """
-        self.failures = []
-        prange = (49152, 65535)
-
-        # register Tor to an ephemeral port
-        socksport = random.randint(prange[0], prange[1])
-        # Keep on trying to get a new port if the chosen one is already
-        # taken.
-        while not self.free_port(socksport):
-            socksport = random.randint(prange[0], prange[1])
-        controlport = random.randint(prange[0], prange[1])
-        while not self.free_port(controlport):
-            controlport = random.randint(prange[0], prange[1])
-
-        randomname = "tor_"+str(random.randint(0, 424242424242))
-        datadir = "/tmp/" + randomname
-        if bridge.startswith("obfs://"):
-            obfsbridge = bridge.split("//")[1]
-
-            self.logger.debug("Genearting torrc file for obfs bridge")
-            torrc = """SocksPort %s
-UseBridges 1
-Bridge obfs2 %s
-DataDirectory %s
-ClientTransportPlugin obfs2 exec /usr/local/bin/obfsproxy --managed
-ControlPort %s
-Log info file %s
-""" % (socksport, obfsbridge, datadir, controlport, os.path.join(datadir,'tor.log'))
-        else:
-            self.logger.debug("Generating torrc file for bridge")
-            if self.tor_greater_than('0.2.3.2'):
-
-                torrc = """SocksPort %s
-UseBridges 1
-bridge %s
-DataDirectory %s
-usemicrodescriptors 0
-ControlPort %s
-Log info file %s
-""" % (socksport, bridge, datadir, controlport, os.path.join(datadir,'tor.log'))
-            else:
-                torrc = """SocksPort %s
-UseBridges 1
-bridge %s
-DataDirectory %s
-ControlPort %s
-Log info file %s
-""" % (socksport, bridge, datadir, controlport, os.path.join(datadir,'tor.log'))
-
-        with open(randomname, "wb") as f:
-            f.write(torrc)
-
-        os.mkdir(datadir)
-        return (randomname, datadir, controlport, socksport)
-
-    def parsebridgeinfo(self, output):
-        ret = {}
-        fields = ['router', 'platform', 'opt', 'published', 'uptime', 'bandwidth']
-
-        for x in output.split("\n"):
-            cfield = x.split(' ')
-            if cfield[0] in fields:
-                #not sure if hellais did this on purpose, but this overwrites
-                #the previous entries. For ex, 'opt' has multiple entries and
-                #only the last value is stored
-                ret[cfield[0]] = ' '.join(cfield[1:])
-                if cfield[1] == 'fingerprint':
-                    ret['fingerprint'] = ''.join(cfield[2:])
-        return ret
-
-    #Can't use @torify as it doesn't support concurrency right now
-    def download_file(self, socksport):
-        opener = urllib2.build_opener(SocksiPyHandler(socks.PROXY_TYPE_SOCKS5,
-                                                      '127.0.0.1', int(socksport)))
-
-        time_start=time.time()
-        f = opener.open('http://38.229.72.16/bwauth.torproject.org/256k')
-        f.read()
-        time_end = time.time()
-        print (time_end-time_start)
-        return str(256/(time_end-time_start)) + " KB/s"
-
-    def is_public(self, fp, socksport):
-        opener = urllib2.build_opener(SocksiPyHandler(socks.PROXY_TYPE_SOCKS5,'127.0.0.1',int(socksport)))
-        response = opener.open(str(ONIONOO_URL)+str(fp))
-        reply = json.loads(response.read())
-        if reply['bridges'] or reply['relays']:
-            return True
-        return False
-
-    def connect(self, bridge, timeout=None):
-        bridgeinfo = None
-        bandwidth = None
-        public = None
-        if not timeout:
-            if self.config.tests.tor_bridges_timeout:
-                self.timeout = self.config.tests.tor_bridges_timeout
-            timeout = self.timeout
-        torrc, tordir, controlport, socksport = self.writetorrc(bridge)
-        cmd = ["tor", "-f", torrc]
-
-        tupdate = time.time()
-        debugupdate = time.time()
-
-        try:
-            p = Popen(cmd, stdout=PIPE)
-        except:
-            self.logger.error("Error in starting Tor (do you have tor installed?)")
-
-        self.logger.info("Testing bridge: %s" % bridge)
-        while True:
-            o = ""
-            try:
-                o = p.stdout.read(4096)
-                if o:
-                    self.logger.debug(str(o))
-                if re.search("100%", o):
-                    self.logger.info("Success in connecting to %s" % bridge)
-
-                    print "%s bridge works" % bridge
-                    # print "%s controlport" % controlport
-                    try:
-                        c = TorCtl.connect('127.0.0.1', controlport)
-                        bridgeinfo = self.parsebridgeinfo(c.get_info('dir/server/all')['dir/server/all'])
-                        c.close()
-                    except:
-                        self.logger.error("Error in connecting to Tor Control port")
-
-                    # XXX disable the public checking
-                    #public = self.is_public(bridgeinfo['fingerprint'], socksport)
-                    #self.logger.info("Public: %s" % public)
-
-                    bandwidth = self.download_file(socksport)
-                    self.logger.info("Bandwidth: %s" % bandwidth)
-
-                    try:
-                        p.stdout.close()
-                    except:
-                        self.logger.error("Error in closing stdout FD.")
-
-                    try:
-                        os.unlink(os.path.join(os.getcwd(), torrc))
-                        rmtree(tordir)
-                    except:
-                        self.logger.error("Error in unlinking files.")
-
-                    p.terminate()
-                    return {
-                            'Time': datetime.now(),
-                            'Bridge': bridge,
-                            'Working': True,
-                            'Descriptor': bridgeinfo,
-                            'Calculated bandwidth': bandwidth,
-                            'Public': public
-                            }
-
-                if re.search("%", o):
-                    # Keep updating the timeout if there is progress
-                    self.logger.debug("Updating time...")
-                    tupdate = time.time()
-                    #print o
-                    continue
-
-            except IOError:
-                ex = sys.exc_info()[1]
-                if ex[0] != errno.EAGAIN:
-                    self.logger.error("Error IOError: EAGAIN")
-                    raise
-                sys.exc_clear()
-                print "In this exception 1"
-
-            try:
-                # Set the timeout for the socket wait
-                ct = timeout-(time.time() - tupdate)
-                socket.wait_read(p.stdout.fileno(), timeout=ct)
-
-            except:
-                lfh = open(os.path.join(tordir, 'tor.log'), 'r')
-                log = lfh.readlines()
-                lfh.close()
-                self.logger.info("%s bridge does not work (%s s timeout)" % (bridge, timeout))
-                print "%s bridge does not work (%s s timeout)" % (bridge, timeout)
-                self.failures.append(bridge)
-                p.stdout.close()
-                os.unlink(os.path.join(os.getcwd(), torrc))
-                rmtree(tordir)
-                p.terminate()
-                return {
-                        'Time': datetime.now(),
-                        'Bridge': bridge,
-                        'Working': False,
-                        'Descriptor': {},
-                        'Log': log
-                        }
-
-    def experiment(self, *a, **kw):
-        # this is just a dirty hack
-        bridge = kw['data']
-        print "Experiment"
-        config = self.config
-
-        return self.connect(bridge)
-
-    def clean(self):
-        for infile in glob.glob('tor_*'):
-            os.remove(infile)
-
-    def print_failures(self):
-        if self.failures:
-            for item in self.failures:
-                print "Offline : %s" % item
-        else:
-            print "All online"
-
-    # For logging TorCtl event msgs
-    #class LogHandler:
-    #def msg(self, severity, message):
-    #   print "[%s] %s"%(severity, message)
-
-def run(ooni, assets=None):
-    """
-    Run the test
-    """
-
-    config = ooni.config
-    urls = []
-
-    bridges = BridgeTAsset(os.path.join(config.main.assetdir, \
-                                        config.tests.tor_bridges))
-
-    bridgelist = [bridges]
-
-    bridget = BridgeT(ooni)
-    ooni.logger.info("Starting bridget test")
-    bridget.run(bridgelist)
-    bridget.print_failures()
-    bridget.clean()
-    ooni.logger.info("Testing completed!")
-
diff --git a/ooni/plugins/new_bridget.py b/ooni/plugins/new_bridget.py
index 3e4db56..b26455e 100644
--- a/ooni/plugins/new_bridget.py
+++ b/ooni/plugins/new_bridget.py
@@ -27,7 +27,7 @@ class bridgetArgs(usage.Options):
 class bridgetTest(OONITest):
     implements(IPlugin, ITest)
 
-    shortName = "bridget"
+    shortName = "newbridget"
     description = "bridget"
     requirements = None
     options = bridgetArgs
diff --git a/ooni/protocols/http.py b/ooni/protocols/http.py
index 2b38f28..09bb9b9 100644
--- a/ooni/protocols/http.py
+++ b/ooni/protocols/http.py
@@ -84,11 +84,7 @@ class HTTPTest(OONITest):
         """
         pass
 
-
-    def experiment(self, args):
-        log.msg("Running experiment")
-        url = self.local_options['url'] if 'url' not in args else args['url']
-
+    def doRequest(self, url):
         d = self.build_request(url)
         def finished(data):
             return data
@@ -97,6 +93,13 @@ class HTTPTest(OONITest):
         d.addCallback(finished)
         return d
 
+    def experiment(self, args):
+        log.msg("Running experiment")
+        url = self.local_options['url'] if 'url' not in args else args['url']
+
+        d = self.doRequest(url)
+        return d
+
     def _cbResponse(self, response):
         self.response['headers'] = list(response.headers.getAllRawHeaders())
         self.response['code'] = response.code
diff --git a/ooni/reporter.py b/ooni/reporter.py
new file mode 100644
index 0000000..0ecf2ea
--- /dev/null
+++ b/ooni/reporter.py
@@ -0,0 +1,13 @@
+from twisted.trial import reporter
+
+class TestResult(reporter.TestResult):
+    """
+    Accumulates the results of several ooni.nettest.TestCases.
+
+    The output format of a TestResult is YAML and it will contain all the basic
+    information that a test result should contain.
+    """
+    def __init__(self):
+        super(TestResult, self).__init__()
+
+
diff --git a/ooni/runner.py b/ooni/runner.py
new file mode 100644
index 0000000..d7caa9d
--- /dev/null
+++ b/ooni/runner.py
@@ -0,0 +1,7 @@
+from twisted.trial import runner
+
+
+class TestLoader(runner.TestLoader):
+    pass
+
+





More information about the tor-commits mailing list