"""
Mutation Abstract Class
============================================
This module defines the abstract base class for mutation methods used in jailbreaking datasets.
These methods transform sequences of text to produce potential adversarial examples, aiding in testing
and strengthening machine learning models against adversarial attacks.

The MutationBase class serves as the foundation for defining specific mutation strategies.
Subclasses implementing specific mutation techniques should override the relevant methods to provide
custom behavior for generating mutated instances.
"""

from abc import ABC, abstractmethod
from typing import List
from jb_datasets.instance import Instance
from jb_datasets.jailbreak_datasets import JailbreakDataset

__all__ = ["MutationBase"]

class MutationBase(ABC):
    """
    An abstract base class for defining mutation strategies that transform a sequence of text to produce
    potential adversarial examples. This class provides the framework for implementing various types of
    text mutations for generating adversarial examples in jailbreak datasets.
    """
    def __call__(self, jailbreak_dataset, *args, **kwargs) -> JailbreakDataset:
        """
        Applies the mutation method to a given jailbreak dataset, generating a new dataset of mutated instances.
        This method provides basic logic for recording parent-child relationships between instances.
        For common 1-to-n mutations, overriding the `get_mutated_instance` method is sufficient.
        For other mutation types, directly overriding the `__call__` method is recommended.

        :param ~JailbreakDataset jailbreak_dataset: The dataset to which the mutation will be applied.
        :return ~JailbreakDataset: A new dataset containing mutated instances.
        """
        new_dataset = []
        for instance in jailbreak_dataset:
            mutated_instance_list = self._get_mutated_instance(instance, *args, **kwargs)
            new_dataset.extend(mutated_instance_list)
        return JailbreakDataset(new_dataset)

    def _get_mutated_instance(self, instance, *args, **kwargs) -> List[Instance]:
        """
        Abstract method to be implemented in subclasses for mutating an instance to generate a list of mutated instances.
        If the mutation method typically generates one or more mutated instances for each input instance,
        this method should be overridden. Otherwise, the `__call__` method should be overridden.

        :param Instance instance: The instance to be mutated.
        :return List[Instance]: A list of mutated instances generated from the input instance.
        """
        raise NotImplementedError
