commit c937439d960ba3cfc77d173615fc029c44e417a7 Author: Arturo Filastò arturo@filasto.net Date: Wed Jan 11 14:35:03 2017 +0000
Add support for deck status --- ooni/agent/scheduler.py | 2 +- ooni/deck/deck.py | 16 +++++++++++++--- ooni/deck/store.py | 3 +-- ooni/director.py | 27 +++++++++++++++++++++++++++ ooni/measurements.py | 16 +++++++++++----- ooni/ui/cli.py | 2 +- ooni/ui/web/server.py | 38 ++++++++++++++++++-------------------- 7 files changed, 72 insertions(+), 32 deletions(-)
diff --git a/ooni/agent/scheduler.py b/ooni/agent/scheduler.py index 4bf058ab..8de3b303 100644 --- a/ooni/agent/scheduler.py +++ b/ooni/agent/scheduler.py @@ -294,7 +294,7 @@ class RunDeck(ScheduledTask): def task(self): deck = deck_store.get(self.deck_id) yield deck.setup() - yield deck.run(self.director) + yield deck.run(self.director, from_schedule=True)
class RefreshDeckList(ScheduledTask): diff --git a/ooni/deck/deck.py b/ooni/deck/deck.py index 07f0799e..fbbb8c3d 100644 --- a/ooni/deck/deck.py +++ b/ooni/deck/deck.py @@ -1,6 +1,7 @@ import os import uuid import errno +import hashlib from copy import deepcopy from string import Template
@@ -82,6 +83,7 @@ class NGDeck(object): self.name = "" self.description = "" self.icon = "" + self.id = None self.schedule = None
self.metadata = {} @@ -102,10 +104,15 @@ class NGDeck(object): def open(self, deck_path, global_options=None): with open(deck_path) as fh: deck_data = yaml.safe_load(fh) + self.id = os.path.basename(deck_path[:-1*len('.yaml')]) self.deck_directory = os.path.abspath(os.path.dirname(deck_path)) self.load(deck_data, global_options)
def load(self, deck_data, global_options=None): + if self.id is None: + # This happens when you load a deck not from a filepath so we + # use the first 16 characters of the SHA256 hexdigest as an ID + self.id = hashlib.sha256(deck_data).hexdigest()[:16] if global_options is not None: self.global_options = normalize_options(global_options)
@@ -209,7 +216,8 @@ class NGDeck(object): ) generate_summary( measurement_dir.child("measurements.njson").path, - measurement_dir.child("summary.json").path + measurement_dir.child("summary.json").path, + deck_id=self.id ) measurement_dir.child("running.pid").remove()
@@ -274,18 +282,20 @@ class NGDeck(object): self._is_setup = True
@defer.inlineCallbacks - def run(self, director): + def run(self, director, from_schedule=False): assert self._is_setup, "You must call setup() before you can run a " \ "deck" if self.requires_tor: yield director.start_tor() yield self.query_bouncer() + director.deckStarted(self.id, from_schedule) for task in self._tasks: if task.skip is True: - log.msg("Skipping running {0}".format(task.id)) + log.debug("Skipping running {0}".format(task.id)) continue if task.type == "ooni": yield self._run_ooni_task(task, director) + director.deckFinished(self.id, from_schedule) self._is_setup = False
diff --git a/ooni/deck/store.py b/ooni/deck/store.py index 6e3e6de4..204d9357 100644 --- a/ooni/deck/store.py +++ b/ooni/deck/store.py @@ -227,11 +227,10 @@ class DeckStore(object): for deck_path in self.available_directory.listdir(): if not deck_path.endswith('.yaml'): continue - deck_id = deck_path[:-1*len('.yaml')] deck = NGDeck( deck_path=self.available_directory.child(deck_path).path ) - new_cache[deck_id] = deck + new_cache[deck.id] = deck self._cache = new_cache self._cache_stale = False
diff --git a/ooni/director.py b/ooni/director.py index 1e55bb67..f7566f5f 100644 --- a/ooni/director.py +++ b/ooni/director.py @@ -70,6 +70,7 @@ class Director(object):
def __init__(self): self.activeNetTests = [] + self.activeDecks = []
self.measurementManager = MeasurementManager() self.measurementManager.director = self @@ -290,6 +291,32 @@ class Director(object): measurement.result = failure return measurement
+ def deckStarted(self, deck_id, from_schedule): + log.debug("Starting {0} ({1})".format(deck_id, + 'scheduled' if from_schedule + else 'user-run')) + self.activeDecks.append((deck_id, from_schedule)) + + def deckFinished(self, deck_id, from_schedule): + log.debug("Finished {0} ({1})".format(deck_id, + 'scheduled' if from_schedule + else 'user-run')) + try: + self.activeDecks.remove((deck_id, from_schedule)) + except ValueError: + log.error("Completed deck {0} is not actually running".format( + deck_id)) + + + def isDeckRunning(self, deck_id, from_schedule): + """ + :param deck_id: the ID of the deck to check if it's running + :param from_schedule: True if we want to know the status of a + scheduled deck run False for user initiated runs. + :return: True if the deck is running False otherwise + """ + return (deck_id, from_schedule) in self.activeDecks + def netTestDone(self, net_test): self.notify(DirectorEvent("success", "Successfully ran test {0}".format( diff --git a/ooni/measurements.py b/ooni/measurements.py index b3b3e0fc..9197ee75 100644 --- a/ooni/measurements.py +++ b/ooni/measurements.py @@ -11,7 +11,7 @@ from ooni.settings import config class MeasurementInProgress(Exception): pass
-class Process(): +class MeasurementTypes(): supported_tests = [ "web_connectivity", "http_requests", @@ -51,14 +51,15 @@ class Process(): result['url'] = entry['input'] return result
-def generate_summary(input_file, output_file): + +def generate_summary(input_file, output_file, deck_id='none'): results = {} with open(input_file) as in_file: for idx, line in enumerate(in_file): entry = json.loads(line.strip()) result = {} - if entry['test_name'] in Process.supported_tests: - result = getattr(Process, entry['test_name'])(entry) + if entry['test_name'] in MeasurementTypes.supported_tests: + result = getattr(MeasurementTypes, entry['test_name'])(entry) result['idx'] = idx if not result.get('url', None): result['url'] = entry['input'] @@ -66,6 +67,7 @@ def generate_summary(input_file, output_file): results['test_start_time'] = entry['test_start_time'] results['country_code'] = entry['probe_cc'] results['asn'] = entry['probe_asn'] + results['deck_id'] = deck_id results['results'] = results.get('results', []) results['results'].append(result)
@@ -73,9 +75,11 @@ def generate_summary(input_file, output_file): json.dump(results, fw) return results
+ class MeasurementNotFound(Exception): pass
+ def get_measurement(measurement_id, compute_size=False): size = -1 measurement_path = FilePath(config.measurements_directory) @@ -114,7 +118,9 @@ def get_measurement(measurement_id, compute_size=False): "keep": keep, "running": running, "stale": stale, - "size": size + "size": size, + # XXX we need the deck ID in here + "deck_id": "none" }
diff --git a/ooni/ui/cli.py b/ooni/ui/cli.py index 162354c1..cf889a3c 100644 --- a/ooni/ui/cli.py +++ b/ooni/ui/cli.py @@ -377,7 +377,7 @@ def runTestWithDirector(director, global_options, url=None, def post_director_start(_): try: yield deck.setup() - yield deck.run(director) + yield deck.run(director, from_schedule=False) except errors.UnableToLoadDeckInput as error: raise defer.failure.Failure(error) except errors.NoReachableTestHelpers as error: diff --git a/ooni/ui/web/server.py b/ooni/ui/web/server.py index efcad146..c8f191c7 100644 --- a/ooni/ui/web/server.py +++ b/ooni/ui/web/server.py @@ -147,7 +147,7 @@ class WebUIAPI(object): app = Klein() # Maximum number in seconds after which to return a result even if no # change happened. - _long_polling_timeout = 5 + _long_polling_timeout = 30 _reactor = reactor _enable_xsrf_protection = True
@@ -171,8 +171,8 @@ class WebUIAPI(object): # We use exponential backoff to trigger retries of the startup of # the director. self._director_startup_retries = 0 - # Maximum delay should be 6 hours - self._director_max_retry_delay = 6*60*60 + # Maximum delay should be 30 minutes + self._director_max_retry_delay = 30*60
self.status_poller = LongPoller( self._long_polling_timeout, _reactor) @@ -368,26 +368,21 @@ class WebUIAPI(object): @xsrf_protect(check=False) @requires_true(attrs=['_director_started', '_is_initialized']) def api_deck_list(self, request): - deck_list = { - 'available': {}, - 'enabled': {} - } + deck_list = {'decks': []} for deck_id, deck in self.director.deck_store.list(): - deck_list['available'][deck_id] = { + deck_list['decks'].append({ + 'id': deck_id, 'name': deck.name, + 'icon': deck.icon, + 'running': self.director.isDeckRunning( + deck_id, from_schedule=False), + 'running_scheduled': self.director.isDeckRunning( + deck_id, from_schedule=True), + 'nettests': [], #XXX 'description': deck.description, 'schedule': deck.schedule, 'enabled': self.director.deck_store.is_enabled(deck_id) - } - - for deck_id, deck in self.director.deck_store.list_enabled(): - deck_list['enabled'][deck_id] = { - 'name': deck.name, - 'description': deck.description, - 'schedule': deck.schedule, - 'enabled': True - } - + }) return self.render_json(deck_list, request)
@app.route('/api/deck/string:deck_id/run', methods=["POST"]) @@ -433,9 +428,12 @@ class WebUIAPI(object): # These are dangling deferreds try: yield deck.setup() - yield deck.run(self.director) + yield deck.run(self.director, from_schedule=False) + self.director_event_poller.notify(DirectorEvent("success", + "Started Deck " + + deck.id)) except: - self.director_event_poller.notify(DirectorEvent("error", + self.director_event_poller.notify(DirectorEvent("error", "Failed to start deck"))
@app.route('/api/nettest/string:test_name/start', methods=["POST"])