Source code for pyrit.orchestrator.single_turn.flip_attack_orchestrator
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT license.
import logging
import pathlib
from typing import Optional
from pyrit.common.path import DATASETS_PATH
from pyrit.models import PromptRequestResponse, SeedPrompt
from pyrit.models.prompt_request_piece import PromptRequestPiece
from pyrit.orchestrator import PromptSendingOrchestrator
from pyrit.prompt_converter.flip_converter import FlipConverter
from pyrit.prompt_target import PromptTarget
from pyrit.score import Scorer
logger = logging.getLogger(__name__)
[docs]
class FlipAttackOrchestrator(PromptSendingOrchestrator):
"""
This orchestrator implements the Flip Attack method found here:
https://arxiv.org/html/2410.02832v1.
Essentially, adds a system prompt to the beginning of the conversation to flip each word in the prompt.
"""
[docs]
def __init__(
self,
objective_target: PromptTarget,
scorers: Optional[list[Scorer]] = None,
batch_size: int = 10,
verbose: bool = False,
) -> None:
"""
Args:
objective_target (PromptTarget): The target for sending prompts.
prompt_converters (list[PromptConverter], Optional): List of prompt converters. These are stacked in
order.
scorers (list[Scorer], Optional): List of scorers to use for each prompt request response, to be
scored immediately after receiving response. Default is None.
batch_size (int, Optional): The (max) batch size for sending prompts. Defaults to 10.
Note: If providing max requests per minute on the prompt_target, this should be set to 1 to
ensure proper rate limit management.\
verbose (bool, Optional): Whether to log debug information. Defaults to False.
"""
super().__init__(
objective_target=objective_target,
prompt_converters=[FlipConverter()],
scorers=scorers,
batch_size=batch_size,
verbose=verbose,
)
# This is sent to the target
system_prompt_path = pathlib.Path(DATASETS_PATH) / "orchestrators" / "flip_attack.yaml"
self.system_prompt = SeedPrompt.from_yaml_file(system_prompt_path).value
system_prompt = PromptRequestResponse(
request_pieces=[
PromptRequestPiece(
role="system",
original_value=self.system_prompt,
)
]
)
super().set_prepended_conversation(prepended_conversation=[system_prompt])
[docs]
async def send_prompts_async( # type: ignore[override]
self,
*,
prompt_list: list[str],
memory_labels: Optional[dict[str, str]] = None,
metadata: Optional[str] = None,
) -> list[PromptRequestResponse]:
"""
Sends the prompts to the prompt target using flip attack.
Args:
prompt_list (list[str]): The list of prompts to be sent.
memory_labels (dict[str, str], Optional): A free-form dictionary of additional labels to apply to the
prompts. Any labels passed in will be combined with self._global_memory_labels with the passed
in labels taking precedence in the case of collisions. Defaults to None.
metadata: Any additional information to be added to the memory entry corresponding to the prompts sent.
Returns:
list[PromptRequestResponse]: The responses from sending the prompts.
"""
# Format follows the paper https://github.com/yueliu1999/FlipAttack/blob/main/result/FlipAttack-gpt-4.json
for i in range(0, len(prompt_list)):
prompt_list[i] = "TASK is '⟪" + prompt_list[i].replace("'", "") + "⟫'"
return await super().send_prompts_async(
prompt_list=prompt_list, prompt_type="text", memory_labels=memory_labels, metadata=metadata
)