2019-09-13 21:12:58 +08:00
|
|
|
import json
|
2019-10-01 15:42:51 +08:00
|
|
|
import logging
|
2022-03-23 03:32:16 +08:00
|
|
|
import time
|
2022-07-28 17:52:42 +08:00
|
|
|
from typing import Sequence, Union
|
2020-04-24 18:19:07 +08:00
|
|
|
|
2019-09-13 21:12:58 +08:00
|
|
|
from bson import json_util
|
|
|
|
|
2022-07-23 01:21:25 +08:00
|
|
|
from common.credentials import Credentials
|
2020-10-07 15:32:33 +08:00
|
|
|
from envs.monkey_zoo.blackbox.island_client.monkey_island_requests import MonkeyIslandRequests
|
2022-07-21 21:43:56 +08:00
|
|
|
from envs.monkey_zoo.blackbox.test_configurations.test_configuration import TestConfiguration
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
SLEEP_BETWEEN_REQUESTS_SECONDS = 0.5
|
2021-04-06 21:19:27 +08:00
|
|
|
MONKEY_TEST_ENDPOINT = "api/test/monkey"
|
|
|
|
TELEMETRY_TEST_ENDPOINT = "api/test/telemetry"
|
|
|
|
LOG_TEST_ENDPOINT = "api/test/log"
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER = logging.getLogger(__name__)
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
|
|
|
|
def avoid_race_condition(func):
|
2022-03-23 03:32:16 +08:00
|
|
|
time.sleep(SLEEP_BETWEEN_REQUESTS_SECONDS)
|
2019-09-13 21:12:58 +08:00
|
|
|
return func
|
|
|
|
|
|
|
|
|
|
|
|
class MonkeyIslandClient(object):
|
|
|
|
def __init__(self, server_address):
|
|
|
|
self.requests = MonkeyIslandRequests(server_address)
|
|
|
|
|
|
|
|
def get_api_status(self):
|
|
|
|
return self.requests.get("api")
|
|
|
|
|
2022-07-28 17:52:42 +08:00
|
|
|
def get_propagation_credentials(self) -> Sequence[Credentials]:
|
|
|
|
response = self.requests.get("api/propagation-credentials")
|
2022-09-01 19:10:54 +08:00
|
|
|
return [Credentials(**credentials) for credentials in response.json()]
|
2021-03-02 21:14:33 +08:00
|
|
|
|
2019-09-13 21:12:58 +08:00
|
|
|
@avoid_race_condition
|
2022-07-21 21:43:56 +08:00
|
|
|
def import_config(self, test_configuration: TestConfiguration):
|
2022-08-19 19:57:50 +08:00
|
|
|
self._set_island_mode()
|
2022-07-27 22:58:46 +08:00
|
|
|
self._import_config(test_configuration)
|
|
|
|
self._import_credentials(test_configuration.propagation_credentials)
|
|
|
|
|
2022-08-19 19:57:50 +08:00
|
|
|
@avoid_race_condition
|
|
|
|
def _set_island_mode(self):
|
|
|
|
if self.requests.put_json("api/island/mode", json="advanced").ok:
|
|
|
|
LOGGER.info("Setting island mode to Custom.")
|
|
|
|
else:
|
|
|
|
LOGGER.error("Failed to set island mode")
|
|
|
|
assert False
|
|
|
|
|
2022-07-27 22:58:46 +08:00
|
|
|
@avoid_race_condition
|
|
|
|
def _import_config(self, test_configuration: TestConfiguration):
|
2022-08-05 21:21:28 +08:00
|
|
|
response = self.requests.put_json(
|
2022-07-23 00:29:41 +08:00
|
|
|
"api/agent-configuration",
|
2022-09-01 15:26:59 +08:00
|
|
|
json=test_configuration.agent_configuration.dict(simplify=True),
|
2022-07-21 21:43:56 +08:00
|
|
|
)
|
2022-07-27 22:58:46 +08:00
|
|
|
if response.ok:
|
|
|
|
LOGGER.info("Configuration is imported.")
|
|
|
|
else:
|
|
|
|
LOGGER.error(f"Failed to import config: {response}")
|
|
|
|
assert False
|
|
|
|
|
|
|
|
@avoid_race_condition
|
|
|
|
def _import_credentials(self, propagation_credentials: Credentials):
|
2022-07-23 01:21:25 +08:00
|
|
|
serialized_propagation_credentials = [
|
2022-09-01 19:10:54 +08:00
|
|
|
Credentials.dict(credentials) for credentials in propagation_credentials
|
2022-07-23 01:21:25 +08:00
|
|
|
]
|
2022-08-05 21:21:43 +08:00
|
|
|
response = self.requests.put_json(
|
2022-07-21 22:15:16 +08:00
|
|
|
"/api/propagation-credentials/configured-credentials",
|
2022-07-23 01:21:57 +08:00
|
|
|
json=serialized_propagation_credentials,
|
2022-07-21 22:15:16 +08:00
|
|
|
)
|
2022-07-27 22:58:46 +08:00
|
|
|
if response.ok:
|
|
|
|
LOGGER.info("Credentials are imported.")
|
|
|
|
else:
|
|
|
|
LOGGER.error(f"Failed to import credentials: {response}")
|
|
|
|
assert False
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
@avoid_race_condition
|
|
|
|
def run_monkey_local(self):
|
2022-07-23 00:34:28 +08:00
|
|
|
response = self.requests.post_json("api/local-monkey", json={"action": "run"})
|
2019-09-13 21:12:58 +08:00
|
|
|
if MonkeyIslandClient.monkey_ran_successfully(response):
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.info("Running the monkey.")
|
2019-09-13 21:12:58 +08:00
|
|
|
else:
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.error("Failed to run the monkey.")
|
2019-09-13 21:12:58 +08:00
|
|
|
assert False
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def monkey_ran_successfully(response):
|
2021-04-06 21:19:27 +08:00
|
|
|
return response.ok and json.loads(response.content)["is_running"]
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
@avoid_race_condition
|
|
|
|
def kill_all_monkeys(self):
|
2022-03-23 03:32:16 +08:00
|
|
|
response = self.requests.post_json(
|
2022-07-23 00:34:28 +08:00
|
|
|
"api/monkey-control/stop-all-agents", json={"kill_time": time.time()}
|
2022-03-23 03:32:16 +08:00
|
|
|
)
|
|
|
|
if response.ok:
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.info("Killing all monkeys after the test.")
|
2019-09-13 21:12:58 +08:00
|
|
|
else:
|
2019-10-01 21:11:53 +08:00
|
|
|
LOGGER.error("Failed to kill all monkeys.")
|
2022-03-23 03:32:16 +08:00
|
|
|
LOGGER.error(response.status_code)
|
|
|
|
LOGGER.error(response.content)
|
2019-09-13 21:12:58 +08:00
|
|
|
assert False
|
|
|
|
|
|
|
|
@avoid_race_condition
|
2022-07-26 22:29:21 +08:00
|
|
|
def reset_island(self):
|
|
|
|
self._reset_agent_configuration()
|
|
|
|
self._reset_simulation_data()
|
|
|
|
self._reset_credentials()
|
|
|
|
self._reset_island_mode()
|
|
|
|
|
|
|
|
def _reset_agent_configuration(self):
|
|
|
|
if self.requests.post("api/reset-agent-configuration", data=None).ok:
|
|
|
|
LOGGER.info("Resetting agent-configuration after the test.")
|
2019-09-13 21:12:58 +08:00
|
|
|
else:
|
2022-07-26 22:29:21 +08:00
|
|
|
LOGGER.error("Failed to reset agent configuration.")
|
|
|
|
assert False
|
|
|
|
|
|
|
|
def _reset_simulation_data(self):
|
|
|
|
if self.requests.post("api/clear-simulation-data", data=None).ok:
|
|
|
|
LOGGER.info("Clearing simulation data.")
|
|
|
|
else:
|
|
|
|
LOGGER.error("Failed to clear simulation data")
|
|
|
|
assert False
|
|
|
|
|
|
|
|
def _reset_credentials(self):
|
2022-08-05 21:34:57 +08:00
|
|
|
if self.requests.put_json("api/propagation-credentials/configured-credentials", json=[]).ok:
|
2022-07-26 22:29:21 +08:00
|
|
|
LOGGER.info("Resseting configured credentials after the test.")
|
|
|
|
else:
|
|
|
|
LOGGER.error("Failed to reset configured credentials")
|
|
|
|
assert False
|
|
|
|
|
|
|
|
def _reset_island_mode(self):
|
2022-08-05 21:35:14 +08:00
|
|
|
if self.requests.put_json("api/island/mode", json="unset").ok:
|
2022-08-02 14:52:53 +08:00
|
|
|
LOGGER.info("Resetting island mode after the test.")
|
2022-07-26 22:29:21 +08:00
|
|
|
else:
|
|
|
|
LOGGER.error("Failed to reset island mode")
|
2019-09-13 21:12:58 +08:00
|
|
|
assert False
|
|
|
|
|
|
|
|
def find_monkeys_in_db(self, query):
|
2019-10-01 15:42:51 +08:00
|
|
|
if query is None:
|
|
|
|
raise TypeError
|
2021-04-06 21:19:27 +08:00
|
|
|
response = self.requests.get(
|
|
|
|
MONKEY_TEST_ENDPOINT, MonkeyIslandClient.form_find_query_for_request(query)
|
|
|
|
)
|
2019-10-01 15:42:51 +08:00
|
|
|
return MonkeyIslandClient.get_test_query_results(response)
|
|
|
|
|
2021-03-08 17:13:31 +08:00
|
|
|
def find_telems_in_db(self, query: dict):
|
|
|
|
if query is None:
|
|
|
|
raise TypeError
|
2021-04-06 21:19:27 +08:00
|
|
|
response = self.requests.get(
|
|
|
|
TELEMETRY_TEST_ENDPOINT, MonkeyIslandClient.form_find_query_for_request(query)
|
|
|
|
)
|
2021-03-08 17:13:31 +08:00
|
|
|
return MonkeyIslandClient.get_test_query_results(response)
|
|
|
|
|
2019-10-01 15:42:51 +08:00
|
|
|
def get_all_monkeys_from_db(self):
|
2021-04-06 21:19:27 +08:00
|
|
|
response = self.requests.get(
|
|
|
|
MONKEY_TEST_ENDPOINT, MonkeyIslandClient.form_find_query_for_request(None)
|
|
|
|
)
|
2019-10-01 15:42:51 +08:00
|
|
|
return MonkeyIslandClient.get_test_query_results(response)
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
def find_log_in_db(self, query):
|
2021-04-06 21:19:27 +08:00
|
|
|
response = self.requests.get(
|
|
|
|
LOG_TEST_ENDPOINT, MonkeyIslandClient.form_find_query_for_request(query)
|
|
|
|
)
|
2019-10-01 15:42:51 +08:00
|
|
|
return MonkeyIslandClient.get_test_query_results(response)
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
@staticmethod
|
2021-03-08 17:13:31 +08:00
|
|
|
def form_find_query_for_request(query: Union[dict, None]) -> dict:
|
2021-04-06 21:19:27 +08:00
|
|
|
return {"find_query": json_util.dumps(query)}
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_test_query_results(response):
|
2021-04-06 21:19:27 +08:00
|
|
|
return json.loads(response.content)["results"]
|
2019-09-13 21:12:58 +08:00
|
|
|
|
|
|
|
def is_all_monkeys_dead(self):
|
2021-04-06 21:19:27 +08:00
|
|
|
query = {"dead": False}
|
2019-09-13 21:12:58 +08:00
|
|
|
return len(self.find_monkeys_in_db(query)) == 0
|