commit b2118234e7b85ad4381ef2caddb9ec37e131fba4 Author: Arturo Filastò arturo@filasto.net Date: Fri Apr 1 19:48:14 2016 +0200
Add support for starting tests from the HTTP API --- ooni/director.py | 2 +- ooni/web/resources.py | 172 ++++++++++++++++++++++++++++++++++++++++++++++++++ ooni/web/root.py | 40 ++++++++++++ ooni/webui.py | 57 +++++++++++++++++ 4 files changed, 270 insertions(+), 1 deletion(-)
diff --git a/ooni/director.py b/ooni/director.py index d6503f2..9897ffd 100644 --- a/ooni/director.py +++ b/ooni/director.py @@ -134,7 +134,7 @@ class Director(object): elif config.tor.control_port and config.tor_state is None: yield connect_to_control_port()
- if config.global_options['no-geoip']: + if config.global_options.get('no-geoip'): aux = [False] if config.global_options.get('annotations') is not None: annotations = [k.lower() for k in config.global_options['annotations'].keys()] diff --git a/ooni/web/resources.py b/ooni/web/resources.py new file mode 100644 index 0000000..336fdc0 --- /dev/null +++ b/ooni/web/resources.py @@ -0,0 +1,172 @@ +import json +from twisted.web import resource +from twisted.python import usage + +from ooni import errors +from ooni.nettest import NetTestLoader + + +class WuiResource(resource.Resource): + isLeaf = True + XSRF_HEADER = 'X-XSRF-TOKEN' + # XXX set this to true when stable version + XSRF_PROTECTION = False + + def __init__(self, director): + self.director = director + resource.Resource.__init__(self) + + def check_xsrf(self, request): + if self.XSRF_PROTECTION is False: + return True + if request.requestHeaders.hasHeader(self.XSRF_HEADER): + return True + return False + + def render(self, request): + if not self.check_xsrf(request): + obj = { + 'error_code': 400, + 'error_message': ('Missing cross site request forgery ' + 'header '{}''.format(self.XSRF_HEADER)) + } + request.setResponseCode(403) + return self.render_json(obj, request) + obj = resource.Resource.render(self, request) + return self.render_json(obj, request) + + def render_json(self, obj, request): + json_string = json.dumps(obj) + "\n" + request.setHeader('Content-Type', 'application/json') + request.setHeader('Content-Length', len(json_string)) + return json_string + + +class DecksGenerate(WuiResource): + def render_GET(self, request): + return {"generate": "deck"} + + +class DecksStart(WuiResource): + def __init__(self, director, deck_name): + WuiResource.__init__(self, director) + self.deck_name = deck_name + + def render_GET(self, request): + return {"start": self.deck_name} + + +class DecksStop(WuiResource): + def __init__(self, director, deck_id): + WuiResource.__init__(self, director) + self.deck_id = deck_id + + def render_GET(self, request): + return {"stop": self.deck_id} + + +class DecksStatus(WuiResource): + def __init__(self, director, deck_name): + WuiResource.__init__(self, director) + self.deck_name = deck_name + + def render_GET(self, request): + return {"deck": self.deck_name} + + +class DecksList(WuiResource): + def render_GET(self, request): + return {"deck": "list"} + + +def getNetTestLoader(test_options, test_file): + """ + Args: + test_options: (dict) containing as keys the option names. + + test_file: (string) the path to the test_file to be run. + Returns: + an instance of :class:`ooni.nettest.NetTestLoader` with the specified + test_file and the specified options. + """ + options = [] + for k, v in test_options.items(): + options.append('--'+k) + options.append(v) + + net_test_loader = NetTestLoader(options, + test_file=test_file) + return net_test_loader + +class TestsStart(WuiResource): + def __init__(self, director, test_name): + WuiResource.__init__(self, director) + self.test_name = test_name + + def render_POST(self, request): + try: + net_test = self.director.netTests[self.test_name] + except KeyError: + request.setResponseCode(500) + return { + 'error_code': 500, + 'error_message': 'Could not find the specified test' + } + test_options = json.load(request.content) + net_test_loader = getNetTestLoader(test_options, net_test['path']) + try: + net_test_loader.checkOptions() + # XXX we actually want to generate the report_filename in a smart + # way so that we can know where it is located and learn the results + # of the measurement. + report_filename = None + self.director.startNetTest(net_test_loader, report_filename) + except errors.MissingRequiredOption, option_name: + request.setResponseCode(500) + return { + 'error_code': 501, + 'error_message': ('Missing required option: ' + ''{}''.format(option_name)) + } + except usage.UsageError: + request.setResponseCode(500) + return { + 'error_code': 502, + 'error_message': 'Error in parsing options' + } + except errors.InsufficientPrivileges: + request.setResponseCode(500) + return { + 'error_code': 503, + 'error_message': 'Insufficient priviledges' + } + + return {"deck": "list"} + + +class TestsStop(WuiResource): + def __init__(self, director, test_id): + WuiResource.__init__(self, director) + self.test_id = test_id + + def render_GET(self, request): + return {"deck": "list"} + + +class TestsStatus(WuiResource): + def __init__(self, director, test_id): + WuiResource.__init__(self, director) + self.test_id = test_id + + def render_GET(self, request): + return {"deck": "list"} + + +class TestsList(WuiResource): + def render_GET(self, request): + return self.director.netTests + + +class Results(WuiResource): + def render_GET(self, request): + return {"result": "bar"} diff --git a/ooni/web/root.py b/ooni/web/root.py new file mode 100644 index 0000000..7ceba4a --- /dev/null +++ b/ooni/web/root.py @@ -0,0 +1,40 @@ +import os +import re +from twisted.web import resource, static + +from .resources import DecksGenerate, DecksStart, DecksStop +from .resources import DecksStatus, DecksList, TestsStart +from .resources import TestsStop, TestsStatus, TestsList +from .resources import Results + + +class OONIProbeWebRoot(resource.Resource): + routes = [ + ('^/decks/generate$', DecksGenerate), + ('^/decks/(.*)/start$', DecksStart), + ('^/decks/(.*)/stop$', DecksStop), + ('^/decks/(.*)$', DecksStatus), + ('^/decks$', DecksList), + ('^/tests/(.*)/start$', TestsStart), + ('^/tests/(.*)/stop$', TestsStop), + ('^/tests/(.*)$', TestsStatus), + ('^/tests$', TestsList), + ('^/results$', Results) + ] + + def __init__(self, config, director): + resource.Resource.__init__(self) + + self._director = director + self._config = config + self._route_map = map(lambda x: (re.compile(x[0]), x[1]), self.routes) + + wui_directory = os.path.join(self._config.data_directory, 'ui', 'app') + self._static = static.File(wui_directory) + + def getChild(self, path, request): + for route, r in self._route_map: + match = route.search(request.path) + if match: + return r(self._director, *match.groups()) + return self._static.getChild(path, request) diff --git a/ooni/webui.py b/ooni/webui.py new file mode 100644 index 0000000..e2fe450 --- /dev/null +++ b/ooni/webui.py @@ -0,0 +1,57 @@ +import os + +from twisted.scripts import twistd +from twisted.python import usage +from twisted.internet import reactor +from twisted.web import server +from twisted.application import service + +from ooni.web.root import OONIProbeWebRoot +from ooni.settings import config +from ooni.director import Director +from ooni.utils import log + +class WebUI(service.MultiService): + portNum = 8822 + def startService(self): + service.MultiService.startService(self) + config.set_paths() + config.initialize_ooni_home() + config.read_config_file() + def _started(res): + log.msg("Director started") + root = server.Site(OONIProbeWebRoot(config, director)) + self._port = reactor.listenTCP(self.portNum, root) + director = Director() + d = director.start() + d.addCallback(_started) + d.addErrback(self._startupFailed) + + def _startupFailed(self, err): + log.err("Failed to start the director") + log.exception(err) + os.abort() + + def stopService(self): + if self._port: + self._port.stopListening() + +class StartOoniprobeWebUIPlugin: + tapname = "ooniprobe" + def makeService(self, so): + return WebUI() + +class MyTwistdConfig(twistd.ServerOptions): + subCommands = [("StartOoniprobeWebUI", None, usage.Options, "ooniprobe web ui")] + +def start(): + twistd_args = ["--nodaemon"] + twistd_config = MyTwistdConfig() + twistd_args.append("StartOoniprobeWebUI") + try: + twistd_config.parseOptions(twistd_args) + except usage.error, ue: + print("ooniprobe: usage error from twistd: {}\n".format(ue)) + twistd_config.loadedPlugins = {"StartOoniprobeWebUI": StartOoniprobeWebUIPlugin()} + twistd.runApp(twistd_config) + return 0