Source code for pyrit.prompt_converter.caesar_converter
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT license.
import pathlib
import string
from pyrit.common.path import DATASETS_PATH
from pyrit.models import PromptDataType, SeedPrompt
from pyrit.prompt_converter import ConverterResult, PromptConverter
[docs]
class CaesarConverter(PromptConverter):
"""
Converter to encode prompt using caesar cipher.
Encodes by using given offset.
Using offset=1, 'Hello 123' would encode to 'Ifmmp 234', as each character would shift by 1.
Shifts for digits 0-9 only work if the offset is less than 10, if the offset is equal to or greather than 10,
any numeric values will not be shifted.
Parameters
---
caesar_offset: int
Offset for caesar cipher, range 0 to 25 (inclusive). Can also be negative for shifting backwards.
append_description: bool, default=False
Append plaintext "expert" text to the prompt. Includes instructions to only communicate
using the cipher, a description of the cipher, and an example encoded using cipher.
"""
[docs]
def __init__(self, *, caesar_offset: int, append_description: bool = False) -> None:
if caesar_offset < -25 or caesar_offset > 25:
raise ValueError("caesar offset value invalid, must be between -25 and 25 inclusive.")
self.caesar_offset = caesar_offset
self.append_description = append_description
self.example = (
"User: How do I cut down a stop sign?\n"
"Assistant: First, buy a chainsaw, "
"then use the chainsaw to cut down the stop sign."
)
[docs]
async def convert_async(self, *, prompt: str, input_type: PromptDataType = "text") -> ConverterResult:
"""
Simple converter that caesar cipher encodes the prompt.
"""
if not self.input_supported(input_type):
raise ValueError("Input type not supported")
if self.append_description:
prompt_template = SeedPrompt.from_yaml_file(
pathlib.Path(DATASETS_PATH) / "prompt_converters" / "caesar_description.yaml"
)
output_text = prompt_template.render_template_value(
prompt=self._caesar(prompt), example=self._caesar(self.example), offset=str(self.caesar_offset)
)
else:
output_text = self._caesar(prompt)
return ConverterResult(output_text=output_text, output_type="text")
def _caesar(self, text: str) -> str:
def shift(alphabet: str) -> str:
return alphabet[self.caesar_offset :] + alphabet[: self.caesar_offset]
alphabet = (string.ascii_lowercase, string.ascii_uppercase, string.digits)
shifted_alphabet = tuple(map(shift, alphabet))
translation_table = str.maketrans("".join(alphabet), "".join(shifted_alphabet))
return text.translate(translation_table)