Source code for boofuzz.blocks.repeat

from .. import helpers
from ..fuzzable import Fuzzable
from ..protocol_session_reference import ProtocolSessionReference


[docs] class Repeat(Fuzzable): """Repeat the rendered contents of the specified block cycling from min_reps to max_reps counting by step. By default renders to nothing. This block modifier is useful for fuzzing overflows in table entries. This block modifier MUST come after the block it is being applied to. :param name: Name, for referencing later. Names should always be provided, but if not, a default name will be given, defaults to None :type name: str, optional :param block_name: Name of block to repeat :type block_name: str, optional :param request: Request this block belongs to, defaults to None :type request: boofuzz.Request, optional :param min_reps: Minimum number of block repetitions, defaults to 0 :type min_reps: int, optional :param max_reps: Maximum number of block repetitions, defaults to None :type max_reps: int, optional :param step: Step count between min and max reps, defaults to 1 :type step: int, optional :param variable: Repetitions will be derived from this variable, disables fuzzing, defaults to None :type variable: Boofuzz Integer Primitive, optional :param default_value: Value used when the element is not being fuzzed - should typically represent a valid value, defaults to None :type default_value: Raw :param fuzzable: Enable/disable fuzzing of this block, defaults to true :type fuzzable: bool, optional """ def __init__( self, name=None, block_name=None, request=None, min_reps=0, max_reps=25, step=1, variable=None, default_value=None, *args, **kwargs ): if default_value is None: if variable is not None: default_value = ProtocolSessionReference(name=variable, default_value=0) else: default_value = 0 super(Repeat, self).__init__(name=name, default_value=default_value, *args, **kwargs) self.block_name = block_name self.request = request self.min_reps = min_reps self.max_reps = max_reps self.step = step self._value = b"" self._original_value = b"" # default to nothing! self._rendered = b"" # rendered value self._fuzz_complete = False # flag if this primitive has been completely fuzzed self._fuzz_library = [] # library of static fuzz heuristics to cycle through. self._mutant_index = 0 # current mutation number self.current_reps = min_reps # current number of repetitions if self.max_reps is not None and self.request is not None and self.block_name is not None: self._fuzz_library = list(range(self.min_reps, self.max_reps + 1, self.step)) def mutations(self, default_value): for fuzzed_reps_number in self._fuzz_library: yield fuzzed_reps_number def num_mutations(self, default_value): """ Determine the number of repetitions we will be making. @rtype: int @return: Number of mutated forms this primitive can take. :param default_value: """ return len(self._fuzz_library) def encode(self, value, mutation_context): return value * self._get_child_data(mutation_context=mutation_context) def _get_child_data(self, mutation_context): if self.request is not None and self.block_name is not None: _rendered = self.request.resolve_name(self.context_path, self.block_name).render( mutation_context=mutation_context ) else: _rendered = "" return helpers.str_to_bytes(_rendered) def __repr__(self): return "<%s %s>" % (self.__class__.__name__, self._name) def __len__(self): return ( self.current_reps * len(self.request.names[self.block_name]) if self.block_name is not None and self.request is not None else 0 )