Source code for volatility3.framework.interfaces.plugins

# This file is Copyright 2019 Volatility Foundation and licensed under the Volatility Software License 1.0
# which is available at https://www.volatilityfoundation.org/license/vsl-v1.0
#
"""Plugins are the `functions` of the volatility framework.

They are called and carry out some algorithms on data stored in layers
using objects constructed from symbols.
"""

# Configuration interfaces must be imported separately, since we're part of interfaces and can't import ourselves
import io
import logging
import os
from abc import ABCMeta, abstractmethod
from typing import List, Tuple, Type

from volatility3.framework import exceptions, constants, interfaces

vollog = logging.getLogger(__name__)


[docs]class FileHandlerInterface(io.RawIOBase): """Class for storing Files in the plugin as a means to output a file when necessary. This can be used as ContextManager that will close/produce the file automatically when exiting the context block """ def __init__(self, filename: str) -> None: """Creates a FileHandler Args: filename: The requested name of the filename for the data """ self._preferred_filename = None self.preferred_filename = filename super().__init__() @property def preferred_filename(self): """The preferred filename to save the data to. Until this file has been written, this value may not be the final filename the data is written to. """ return self._preferred_filename @preferred_filename.setter def preferred_filename(self, filename: str): """Sets the preferred filename""" if self.closed: raise IOError("FileHandler name cannot be changed once closed") if not isinstance(filename, str): raise TypeError("FileHandler preferred filenames must be strings") if os.path.sep in filename: raise ValueError("FileHandler filenames cannot contain path separators") self._preferred_filename = filename
[docs] @abstractmethod def close(self): """Method that commits the file and fixes the final filename for use"""
[docs] @staticmethod def sanitize_filename(filename: str) -> str: """Sanititizes the filename to ensure only a specific whitelist of characters is allowed through""" allowed = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789.- ()[]{}!$%^:#~?<>,|" result = "" for char in filename: if char in allowed: result += char else: result += "?" return result
def __enter__(self): return self def __exit__(self, exc_type, exc_value, traceback): if exc_type is None and exc_value is None and traceback is None: self.close() else: vollog.warning( f"File {self._preferred_filename} could not be written: {str(exc_value)}" ) self.close()
# # Plugins # - Take in relevant number of TranslationLayers (of specified type) # - Outputs TreeGrid # # Should the plugin handle constructing the translation layers from the filenames or should the library have routines for it? # Outwardly, the user specifies an OS, version, architecture triple and images. # The UI checks the plugin against the OS/Version/Arch triple # The UI constructs the TranslationLayers and names them according to the plugin's input layer names # The UI constructs the appropriate default symbol spaces # The plugin accepts the context and modifies as necessary # The plugin runs and produces a TreeGrid output
[docs]class PluginInterface( interfaces.configuration.ConfigurableInterface, interfaces.configuration.VersionableInterface, metaclass=ABCMeta, ): """Class that defines the basic interface that all Plugins must maintain. The constructor must only take a `context` and `config_path`, so that plugins can be launched automatically. As such all configuration information must be provided through the requirements and configuration information in the context it is passed. """ # Be careful with inheritance around this (We default to requiring a version which doesn't exist, so it must be set) _required_framework_version: Tuple[int, int, int] = (0, 0, 0) """The _version variable is a quick way for plugins to define their current interface, it should follow SemVer rules""" def __init__( self, context: interfaces.context.ContextInterface, config_path: str, progress_callback: constants.ProgressCallback = None, ) -> None: """ Args: context: The context that the plugin will operate within config_path: The path to configuration data within the context configuration data progress_callback: A callable that can provide feedback at progress points """ super().__init__(context, config_path) self._progress_callback = progress_callback or (lambda f, s: None) # Plugins self validate on construction, it makes it more difficult to work with them, but then # the validation doesn't need to be repeated over and over again by externals if self.unsatisfied(context, config_path): vollog.warning("Plugin failed validation") raise exceptions.PluginRequirementException( "The plugin configuration failed to validate" ) # Populate any optional defaults for requirement in self.get_requirements(): if requirement.name not in self.config: self.config[requirement.name] = requirement.default self._file_handler: Type[FileHandlerInterface] = FileHandlerInterface @property def open(self): """Returns a context manager and thus can be called like open""" return self._file_handler
[docs] def set_open_method(self, handler: Type[FileHandlerInterface]) -> None: """Sets the file handler to be used by this plugin.""" if not issubclass(handler, FileHandlerInterface): raise ValueError("FileHandler must be a subclass of FileHandlerInterface") self._file_handler = handler
[docs] @classmethod def get_requirements(cls) -> List[interfaces.configuration.RequirementInterface]: """Returns a list of Requirement objects for this plugin.""" return super().get_requirements()
[docs] @abstractmethod def run(self) -> interfaces.renderers.TreeGrid: """Executes the functionality of the code. .. note:: This method expects `self.validate` to have been called to ensure all necessary options have been provided Returns: A TreeGrid object that can then be passed to a Renderer. """