2019-10-07 15:39:11 +08:00
|
|
|
import json
|
2019-09-11 17:39:28 +08:00
|
|
|
from time import sleep
|
|
|
|
|
2019-10-01 15:42:51 +08:00
|
|
|
import logging
|
|
|
|
|
2019-09-11 17:39:28 +08:00
|
|
|
from envs.monkey_zoo.blackbox.utils.test_timer import TestTimer
|
|
|
|
|
2019-10-01 21:11:53 +08:00
|
|
|
MAX_TIME_FOR_MONKEYS_TO_DIE = 5 * 60
|
2019-09-13 21:12:58 +08:00
|
|
|
WAIT_TIME_BETWEEN_REQUESTS = 10
|
2019-09-24 20:38:16 +08:00
|
|
|
TIME_FOR_MONKEY_PROCESS_TO_FINISH = 40
|
2019-10-01 15:42:51 +08:00
|
|
|
DELAY_BETWEEN_ANALYSIS = 3
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER = logging.getLogger(__name__)
|
2019-09-11 17:39:28 +08:00
|
|
|
|
|
|
|
|
|
|
|
class BasicTest(object):
|
|
|
|
|
2019-10-01 15:42:51 +08:00
|
|
|
def __init__(self, name, island_client, config_parser, analyzers, timeout, log_handler):
|
2019-09-11 17:39:28 +08:00
|
|
|
self.name = name
|
|
|
|
self.island_client = island_client
|
2019-09-17 14:17:29 +08:00
|
|
|
self.config_parser = config_parser
|
2019-09-11 17:39:28 +08:00
|
|
|
self.analyzers = analyzers
|
|
|
|
self.timeout = timeout
|
2019-10-01 15:42:51 +08:00
|
|
|
self.log_handler = log_handler
|
2019-09-11 17:39:28 +08:00
|
|
|
|
|
|
|
def run(self):
|
2019-10-07 15:39:11 +08:00
|
|
|
LOGGER.info("Uploading configuration:\n{}".format(json.dumps(self.config_parser.config_json, indent=2)))
|
2019-09-17 14:17:29 +08:00
|
|
|
self.island_client.import_config(self.config_parser.config_raw)
|
|
|
|
self.print_test_starting_info()
|
2019-09-11 17:39:28 +08:00
|
|
|
try:
|
|
|
|
self.island_client.run_monkey_local()
|
|
|
|
self.test_until_timeout()
|
|
|
|
finally:
|
|
|
|
self.island_client.kill_all_monkeys()
|
2019-09-13 21:12:58 +08:00
|
|
|
self.wait_until_monkeys_die()
|
2019-09-17 14:17:29 +08:00
|
|
|
self.wait_for_monkey_process_to_finish()
|
|
|
|
self.parse_logs()
|
2019-09-11 17:39:28 +08:00
|
|
|
self.island_client.reset_env()
|
|
|
|
|
2019-09-17 14:17:29 +08:00
|
|
|
def print_test_starting_info(self):
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.info("Started {} test".format(self.name))
|
|
|
|
LOGGER.info("Machines participating in test:")
|
|
|
|
LOGGER.info(" ".join(self.config_parser.get_ips_of_targets()))
|
|
|
|
print("")
|
2019-09-17 14:17:29 +08:00
|
|
|
|
2019-09-11 17:39:28 +08:00
|
|
|
def test_until_timeout(self):
|
|
|
|
timer = TestTimer(self.timeout)
|
2019-10-01 15:42:51 +08:00
|
|
|
while not timer.is_timed_out():
|
2019-09-11 17:39:28 +08:00
|
|
|
if self.all_analyzers_pass():
|
|
|
|
self.log_success(timer)
|
|
|
|
return
|
2019-10-01 15:42:51 +08:00
|
|
|
sleep(DELAY_BETWEEN_ANALYSIS)
|
2019-09-11 17:39:28 +08:00
|
|
|
self.log_failure(timer)
|
|
|
|
assert False
|
|
|
|
|
|
|
|
def log_success(self, timer):
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.info(self.get_analyzer_logs())
|
|
|
|
LOGGER.info("{} test passed, time taken: {:.1f} seconds.".format(self.name, timer.get_time_taken()))
|
2019-09-11 17:39:28 +08:00
|
|
|
|
|
|
|
def log_failure(self, timer):
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.info(self.get_analyzer_logs())
|
|
|
|
LOGGER.error("{} test failed because of timeout. Time taken: {:.1f} seconds.".format(self.name,
|
|
|
|
timer.get_time_taken()))
|
2019-09-11 17:39:28 +08:00
|
|
|
|
|
|
|
def all_analyzers_pass(self):
|
|
|
|
for analyzer in self.analyzers:
|
|
|
|
if not analyzer.analyze_test_results():
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_analyzer_logs(self):
|
|
|
|
log = ""
|
|
|
|
for analyzer in self.analyzers:
|
2019-10-01 21:11:53 +08:00
|
|
|
log += "\n" + analyzer.log.get_contents()
|
2019-09-11 17:39:28 +08:00
|
|
|
return log
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
def wait_until_monkeys_die(self):
|
|
|
|
time_passed = 0
|
|
|
|
while not self.island_client.is_all_monkeys_dead() and time_passed < MAX_TIME_FOR_MONKEYS_TO_DIE:
|
|
|
|
sleep(WAIT_TIME_BETWEEN_REQUESTS)
|
|
|
|
time_passed += WAIT_TIME_BETWEEN_REQUESTS
|
2019-10-07 17:20:19 +08:00
|
|
|
LOGGER.debug("Waiting for all monkeys to die. Time passed: {}".format(time_passed))
|
2019-09-13 21:12:58 +08:00
|
|
|
if time_passed > MAX_TIME_FOR_MONKEYS_TO_DIE:
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.error("Some monkeys didn't die after the test, failing")
|
2019-09-13 21:12:58 +08:00
|
|
|
assert False
|
2019-09-17 14:17:29 +08:00
|
|
|
|
|
|
|
def parse_logs(self):
|
2019-10-07 17:20:19 +08:00
|
|
|
LOGGER.info("Parsing test logs:")
|
2019-10-01 15:42:51 +08:00
|
|
|
self.log_handler.parse_test_logs()
|
2019-09-17 14:17:29 +08:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def wait_for_monkey_process_to_finish():
|
2019-10-01 15:42:51 +08:00
|
|
|
"""
|
|
|
|
There is a time period when monkey is set to dead, but the process is still closing.
|
|
|
|
If we try to launch monkey during that time window monkey will fail to start, that's
|
|
|
|
why test needs to wait a bit even after all monkeys are dead.
|
|
|
|
"""
|
2019-09-17 14:17:29 +08:00
|
|
|
sleep(TIME_FOR_MONKEY_PROCESS_TO_FINISH)
|