From 89436a4cd9d89355ccf685371ed1f3e57c14c63d Mon Sep 17 00:00:00 2001 From: VakarisZ Date: Wed, 1 Dec 2021 12:11:38 +0200 Subject: [PATCH] Agent: remove behavioral methods from monkey.py and leave only setup/teardown related code Behavior is handled by master, monkey.py should only setup/teardown the agent --- monkey/infection_monkey/monkey.py | 381 +----------------- .../exceptions/planned_shutdown_exception.py | 2 - 2 files changed, 21 insertions(+), 362 deletions(-) delete mode 100644 monkey/infection_monkey/utils/exceptions/planned_shutdown_exception.py diff --git a/monkey/infection_monkey/monkey.py b/monkey/infection_monkey/monkey.py index bfbf270dc..9b36ba59e 100644 --- a/monkey/infection_monkey/monkey.py +++ b/monkey/infection_monkey/monkey.py @@ -3,77 +3,53 @@ import logging import os import subprocess import sys -import time -from threading import Thread import infection_monkey.tunnel as tunnel from common.utils.attack_utils import ScanStatus, UsageEnum -from common.utils.exceptions import ExploitingVulnerableMachineError, FailedExploitationError from common.version import get_version from infection_monkey.config import WormConfiguration from infection_monkey.control import ControlClient -from infection_monkey.exploit.HostExploiter import HostExploiter from infection_monkey.master.mock_master import MockMaster from infection_monkey.model import DELAY_DELETE_CMD from infection_monkey.network.firewall import app as firewall -from infection_monkey.network.HostFinger import HostFinger -from infection_monkey.network.network_scanner import NetworkScanner -from infection_monkey.network.tools import get_interface_to_target, is_running_on_island -from infection_monkey.post_breach.post_breach_handler import PostBreach +from infection_monkey.network.tools import is_running_on_island from infection_monkey.puppet.mock_puppet import MockPuppet -from infection_monkey.ransomware.ransomware_payload_builder import build_ransomware_payload -from infection_monkey.system_info import SystemInfoCollector from infection_monkey.system_singleton import SystemSingleton from infection_monkey.telemetry.attack.t1106_telem import T1106Telem from infection_monkey.telemetry.attack.t1107_telem import T1107Telem -from infection_monkey.telemetry.attack.victim_host_telem import VictimHostTelem from infection_monkey.telemetry.messengers.legacy_telemetry_messenger_adapter import ( LegacyTelemetryMessengerAdapter, ) -from infection_monkey.telemetry.scan_telem import ScanTelem from infection_monkey.telemetry.state_telem import StateTelem -from infection_monkey.telemetry.system_info_telem import SystemInfoTelem -from infection_monkey.telemetry.trace_telem import TraceTelem from infection_monkey.telemetry.tunnel_telem import TunnelTelem from infection_monkey.utils.environment import is_windows_os -from infection_monkey.utils.exceptions.planned_shutdown_exception import PlannedShutdownException -from infection_monkey.utils.monkey_dir import ( - create_monkey_dir, - get_monkey_dir_path, - remove_monkey_dir, -) +from infection_monkey.utils.monkey_dir import get_monkey_dir_path, remove_monkey_dir from infection_monkey.utils.monkey_log_path import get_monkey_log_path from infection_monkey.utils.signal_handler import register_signal_handlers from infection_monkey.windows_upgrader import WindowsUpgrader -MAX_DEPTH_REACHED_MESSAGE = "Reached max depth, skipping propagation phase." - - logger = logging.getLogger(__name__) -class InfectionMonkey(object): +class PlannedShutdownError(Exception): + # Raise when we deliberately want to shut down the agent + pass + + +class InfectionMonkey: def __init__(self, args): logger.info("Monkey is initializing...") self._master = MockMaster(MockPuppet(), LegacyTelemetryMessengerAdapter()) - self._keep_running = False - self._exploited_machines = set() - self._fail_exploitation_machines = set() self._singleton = SystemSingleton() self._opts = None self._set_arguments(args) self._parent = self._opts.parent self._default_tunnel = self._opts.tunnel self._default_server = self._opts.server + self._default_server_port = None self._set_propagation_depth() self._add_default_server_to_config() - self._network = NetworkScanner() - self._exploiters = None - self._fingerprint = None - self._default_server_port = None - self._upgrading_to_64 = False self._monkey_tunnel = None - self._post_breach_phase = None def _set_arguments(self, args): arg_parser = argparse.ArgumentParser() @@ -108,8 +84,7 @@ class InfectionMonkey(object): def start(self): if self._is_another_monkey_running(): - logger.info("Another instance of the monkey is already running") - return + raise PlannedShutdownError("Another instance of the monkey is already running.") logger.info("Monkey is starting...") @@ -120,35 +95,15 @@ class InfectionMonkey(object): T1106Telem(ScanStatus.USED, UsageEnum.SINGLETON_WINAPI).send() if InfectionMonkey._is_monkey_alive_by_config(): - logger.error("Monkey marked 'not alive' from configuration.") - return + raise PlannedShutdownError("Monkey marked 'not alive' from configuration.") if InfectionMonkey._is_upgrade_to_64_needed(): self._upgrade_to_64() - return + raise PlannedShutdownError("32 bit Agent can't run on 64 bit system.") self._setup() self._master.start() - def legacy_start(self): - if self._is_another_monkey_running(): - raise Exception("Another instance of the monkey is already running") - try: - logger.info("Monkey is starting...") - - self._legacy_setup() - - self._start_post_breach_async() - - self._start_propagation() - - except PlannedShutdownException: - logger.info( - "A planned shutdown of the Monkey occurred. Logging the reason and finishing " - "execution." - ) - logger.exception("Planned shutdown, reason:") - def _connect_to_island(self): # Sets island's IP and port for monkey to communicate to if not self._is_default_server_set(): @@ -180,33 +135,15 @@ class InfectionMonkey(object): return WindowsUpgrader.should_upgrade() def _upgrade_to_64(self): - self._upgrading_to_64 = True self._singleton.unlock() logger.info("32bit monkey running on 64bit Windows. Upgrading.") WindowsUpgrader.upgrade(self._opts) logger.info("Finished upgrading from 32bit to 64bit.") - def _legacy_upgrade_to_64_if_needed(self): - if WindowsUpgrader.should_upgrade(): - self._upgrading_to_64 = True - self._singleton.unlock() - logger.info("32bit monkey running on 64bit Windows. Upgrading.") - WindowsUpgrader.upgrade(self._opts) - raise PlannedShutdownException("Finished upgrading from 32bit to 64bit.") - def _setup(self): logger.debug("Starting the setup phase.") - if is_running_on_island(): - # TODO: Evaluate also this with ControlClient.should_monkey_run - # WormConfiguration.started_on_island = True - ControlClient.report_start_on_island() - - # TODO: Evaluate should we run this check - # if not ControlClient.should_monkey_run(self._opts.vulnerable_port): - # logger.error("Monkey shouldn't run on current machine " - # "(it will be exploited later with more depth).") - # return False + self._should_run_check_for_performance() if firewall.is_enabled(): firewall.add_firewall_rule() @@ -220,307 +157,31 @@ class InfectionMonkey(object): register_signal_handlers(self._master) - def _legacy_setup(self): - logger.debug("Starting the setup phase.") - - self._keep_running = True - - # Create a dir for monkey files if there isn't one - create_monkey_dir() - - # Sets island's IP and port for monkey to communicate to - self._legacy_set_default_server() - self._set_default_port() - - ControlClient.wakeup(parent=self._parent) - ControlClient.load_control_config() - - if is_windows_os(): - T1106Telem(ScanStatus.USED, UsageEnum.SINGLETON_WINAPI).send() - - InfectionMonkey._legacy_shutdown_by_not_alive_config() - - self._legacy_upgrade_to_64_if_needed() - + def _should_run_check_for_performance(self): + """ + This method implements propagation performance enhancing algorithm that + kicks in if the run was started from the Island. + Should get replaced by other, better performance enhancement solutions + """ if is_running_on_island(): WormConfiguration.started_on_island = True ControlClient.report_start_on_island() if not ControlClient.should_monkey_run(self._opts.vulnerable_port): - raise PlannedShutdownException( - "Monkey shouldn't run on current machine " + raise PlannedShutdownError( + "Monkey shouldn't run on current machine to improve perfomance" "(it will be exploited later with more depth)." ) - if firewall.is_enabled(): - firewall.add_firewall_rule() - - self._monkey_tunnel = ControlClient.create_control_tunnel() - if self._monkey_tunnel: - self._monkey_tunnel.start() - - StateTelem(is_done=False, version=get_version()).send() - TunnelTelem().send() - def _is_another_monkey_running(self): return not self._singleton.try_lock() - def _legacy_set_default_server(self): - """ - Sets the default server for the Monkey to communicate back to. - :raises PlannedShutdownException if couldn't find the server. - """ - if not ControlClient.find_server(default_tunnel=self._default_tunnel): - raise PlannedShutdownException( - "Monkey couldn't find server with {} default tunnel.".format(self._default_tunnel) - ) - self._default_server = WormConfiguration.current_server - logger.debug("default server set to: %s" % self._default_server) - def _set_default_port(self): try: self._default_server_port = self._default_server.split(":")[1] except KeyError: self._default_server_port = "" - @staticmethod - def _legacy_shutdown_by_not_alive_config(): - if not WormConfiguration.alive: - raise PlannedShutdownException("Marked 'not alive' from configuration.") - - def _start_post_breach_async(self): - logger.debug("Starting the post-breach phase asynchronously.") - self._post_breach_phase = Thread(target=InfectionMonkey._start_post_breach_phase) - self._post_breach_phase.start() - - @staticmethod - def _start_post_breach_phase(): - InfectionMonkey._collect_system_info_if_configured() - PostBreach().execute_all_configured() - - @staticmethod - def _collect_system_info_if_configured(): - logger.debug("Calling for system info collection") - try: - system_info_collector = SystemInfoCollector() - system_info = system_info_collector.get_info() - SystemInfoTelem(system_info).send() - except Exception as e: - logger.exception(f"Exception encountered during system info collection: {str(e)}") - - def _start_propagation(self): - if not InfectionMonkey._max_propagation_depth_reached(): - logger.info("Starting the propagation phase.") - logger.debug("Running with depth: %d" % WormConfiguration.depth) - self._propagate() - else: - logger.info("Maximum propagation depth has been reached; monkey will not propagate.") - TraceTelem(MAX_DEPTH_REACHED_MESSAGE).send() - - if self._keep_running and WormConfiguration.alive: - InfectionMonkey._run_ransomware() - - # if host was exploited, before continue to closing the tunnel ensure the exploited - # host had its chance to connect to the tunnel - if len(self._exploited_machines) > 0: - time_to_sleep = WormConfiguration.keep_tunnel_open_time - logger.info( - "Sleeping %d seconds for exploited machines to connect to tunnel", time_to_sleep - ) - time.sleep(time_to_sleep) - - @staticmethod - def _max_propagation_depth_reached(): - return 0 == WormConfiguration.depth - - def _propagate(self): - ControlClient.keepalive() - ControlClient.load_control_config() - - self._network.initialize() - - self._fingerprint = HostFinger.get_instances() - - self._exploiters = HostExploiter.get_classes() - - if not WormConfiguration.alive: - logger.info("Marked not alive from configuration") - - machines = self._network.get_victim_machines( - max_find=WormConfiguration.victims_max_find, - stop_callback=ControlClient.check_for_stop, - ) - for machine in machines: - if ControlClient.check_for_stop(): - break - - for finger in self._fingerprint: - logger.info( - "Trying to get OS fingerprint from %r with module %s", - machine, - finger.__class__.__name__, - ) - try: - finger.get_host_fingerprint(machine) - except BaseException as exc: - logger.error( - "Failed to run fingerprinter %s, exception %s" % finger.__class__.__name__, - str(exc), - ) - - ScanTelem(machine).send() - - # skip machines that we've already exploited - if machine in self._exploited_machines: - logger.debug("Skipping %r - already exploited", machine) - continue - - if self._monkey_tunnel: - self._monkey_tunnel.set_tunnel_for_host(machine) - if self._default_server: - if self._network.on_island(self._default_server): - machine.set_default_server( - get_interface_to_target(machine.ip_addr) - + (":" + self._default_server_port if self._default_server_port else "") - ) - else: - machine.set_default_server(self._default_server) - logger.debug( - "Default server for machine: %r set to %s" % (machine, machine.default_server) - ) - - # Order exploits according to their type - self._exploiters = sorted( - self._exploiters, key=lambda exploiter_: exploiter_.EXPLOIT_TYPE.value - ) - host_exploited = False - for exploiter in [exploiter(machine) for exploiter in self._exploiters]: - if self._try_exploiting(machine, exploiter): - host_exploited = True - VictimHostTelem("T1210", ScanStatus.USED, machine=machine).send() - if exploiter.RUNS_AGENT_ON_SUCCESS: - break # if adding machine to exploited, won't try other exploits - # on it - if not host_exploited: - self._fail_exploitation_machines.add(machine) - VictimHostTelem("T1210", ScanStatus.SCANNED, machine=machine).send() - if not self._keep_running: - break - - if not WormConfiguration.alive: - logger.info("Marked not alive from configuration") - - def _try_exploiting(self, machine, exploiter): - """ - Workflow of exploiting one machine with one exploiter - :param machine: Machine monkey tries to exploit - :param exploiter: Exploiter to use on that machine - :return: True if successfully exploited, False otherwise - """ - if not exploiter.is_os_supported(): - logger.info( - "Skipping exploiter %s host:%r, os %s is not supported", - exploiter.__class__.__name__, - machine, - machine.os, - ) - return False - - logger.info( - "Trying to exploit %r with exploiter %s...", machine, exploiter.__class__.__name__ - ) - - result = False - try: - result = exploiter.exploit_host() - if result: - self._successfully_exploited(machine, exploiter, exploiter.RUNS_AGENT_ON_SUCCESS) - return True - else: - logger.info( - "Failed exploiting %r with exploiter %s", machine, exploiter.__class__.__name__ - ) - except ExploitingVulnerableMachineError as exc: - logger.error( - "Exception while attacking %s using %s: %s", - machine, - exploiter.__class__.__name__, - exc, - ) - self._successfully_exploited(machine, exploiter, exploiter.RUNS_AGENT_ON_SUCCESS) - return True - except FailedExploitationError as e: - logger.info( - "Failed exploiting %r with exploiter %s, %s", - machine, - exploiter.__class__.__name__, - e, - ) - except Exception as exc: - logger.exception( - "Exception while attacking %s using %s: %s", - machine, - exploiter.__class__.__name__, - exc, - ) - finally: - exploiter.send_exploit_telemetry(exploiter.__class__.__name__, result) - return False - - def _successfully_exploited(self, machine, exploiter, RUNS_AGENT_ON_SUCCESS=True): - """ - Workflow of registering successfully exploited machine - :param machine: machine that was exploited - :param exploiter: exploiter that succeeded - """ - if RUNS_AGENT_ON_SUCCESS: - self._exploited_machines.add(machine) - - logger.info("Successfully propagated to %s using %s", machine, exploiter.__class__.__name__) - - # check if max-exploitation limit is reached - if WormConfiguration.victims_max_exploit <= len(self._exploited_machines): - self._keep_running = False - - logger.info("Max exploited victims reached (%d)", WormConfiguration.victims_max_exploit) - - @staticmethod - def _run_ransomware(): - try: - ransomware_payload = build_ransomware_payload(WormConfiguration.ransomware) - ransomware_payload.run_payload() - except Exception as ex: - logger.error(f"An unexpected error occurred while running the ransomware payload: {ex}") - - def legacy_cleanup(self): - logger.info("Monkey cleanup started") - self._keep_running = False - if self._monkey_tunnel: - self._monkey_tunnel.stop() - self._monkey_tunnel.join() - - if self._post_breach_phase: - self._post_breach_phase.join() - - if firewall.is_enabled(): - firewall.remove_firewall_rule() - firewall.close() - - if self._upgrading_to_64: - InfectionMonkey._close_tunnel() - firewall.close() - else: - StateTelem( - is_done=True, version=get_version() - ).send() # Signal the server (before closing the tunnel) - InfectionMonkey._close_tunnel() - firewall.close() - InfectionMonkey._send_log() - self._singleton.unlock() - - InfectionMonkey._self_delete() - logger.info("Monkey is shutting down") - def cleanup(self): logger.info("Monkey cleanup started") try: diff --git a/monkey/infection_monkey/utils/exceptions/planned_shutdown_exception.py b/monkey/infection_monkey/utils/exceptions/planned_shutdown_exception.py deleted file mode 100644 index f0147e1e5..000000000 --- a/monkey/infection_monkey/utils/exceptions/planned_shutdown_exception.py +++ /dev/null @@ -1,2 +0,0 @@ -class PlannedShutdownException(Exception): - pass