762 lines
24 KiB
Java
762 lines
24 KiB
Java
//CipherStreamJava/src/main/java/com/mattrixwv/cipherstream/polysubstitution/Playfair.java
|
|
//Matthew Ellison
|
|
// Created: 07-30-21
|
|
//Modified: 08-11-24
|
|
/*
|
|
Copyright (C) 2024 Mattrixwv
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
package com.mattrixwv.cipherstream.polysubstitution;
|
|
|
|
|
|
import java.util.StringJoiner;
|
|
|
|
import org.slf4j.Logger;
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
import com.mattrixwv.cipherstream.exceptions.InvalidCharacterException;
|
|
import com.mattrixwv.cipherstream.exceptions.InvalidInputException;
|
|
import com.mattrixwv.cipherstream.exceptions.InvalidKeywordException;
|
|
|
|
|
|
/**
|
|
* Represents the Playfair cipher encryption and decryption.
|
|
* The Playfair cipher is a digraph substitution cipher that encrypts pairs of letters.
|
|
*/
|
|
public class Playfair{
|
|
private static final Logger logger = LoggerFactory.getLogger(Playfair.class);
|
|
|
|
/** A class representing the location of a character in the grid */
|
|
protected class CharLocation{
|
|
/** The x location in the grid */
|
|
protected int x;
|
|
/** The y location in the grid */
|
|
protected int y;
|
|
/**
|
|
* Constructs a CharLocation with the specified x and y coordinates.
|
|
*
|
|
* @param x the x-coordinate of the character's location
|
|
* @param y the y-coordinate of the character's location
|
|
*/
|
|
public CharLocation(int x, int y){
|
|
this.x = x;
|
|
this.y = y;
|
|
}
|
|
/**
|
|
* Returns the x-coordinate of the character's location.
|
|
*
|
|
* @return the x-coordinate
|
|
*/
|
|
public int getX(){
|
|
return x;
|
|
}
|
|
/**
|
|
* Returns the y-coordinate of the character's location.
|
|
*
|
|
* @return the y-coordinate
|
|
*/
|
|
public int getY(){
|
|
return y;
|
|
}
|
|
}
|
|
|
|
//?Fields
|
|
/** The message that needs to be encoded/decoded */
|
|
protected String inputString;
|
|
/** The encoded/decoded message */
|
|
protected String outputString;
|
|
/** The keyword used to create the grid */
|
|
protected String keyword;
|
|
/** The grid used to encode/decode the message */
|
|
protected char[][] grid;
|
|
/** The letter that will need to be replaced in the grid and any input string or keyword */
|
|
protected char replaced;
|
|
/** The letter that replaced replaced in the input string or keyword */
|
|
protected char replacer;
|
|
/** The letter that will be placed between double letters in the input string if necessary or to make the string length even */
|
|
protected char doubled;
|
|
//?Settings
|
|
/** Persist capitals in the output string */
|
|
protected boolean preserveCapitals;
|
|
/** Persist whitespace in the output string */
|
|
protected boolean preserveWhitespace;
|
|
/** Persist symbols in the output string */
|
|
protected boolean preserveSymbols;
|
|
|
|
|
|
/**
|
|
* Sets the doubled character.
|
|
*
|
|
* @param doubled the character to be used as the doubled character
|
|
* @throws InvalidCharacterException if the character is not a letter or is invalid
|
|
*/
|
|
protected void setDoubled(char doubled) throws InvalidCharacterException{
|
|
logger.debug("Setting doubled");
|
|
logger.debug("Original character {}", doubled);
|
|
|
|
//Make sure the character is a letter
|
|
logger.debug("Checking letter");
|
|
if(!Character.isAlphabetic(doubled)){
|
|
throw new InvalidCharacterException("The double letter replacement must be a letter");
|
|
}
|
|
|
|
//Make sure the 2 replacers are the same
|
|
logger.debug("Checking same as replacer");
|
|
if(doubled == replacer){
|
|
throw new InvalidCharacterException("The double letter replacement cannot be the same as the regular replacer");
|
|
}
|
|
|
|
//Make sure the double letter replacement isn't the same as the letter being replaced
|
|
logger.debug("Checking same as replaced");
|
|
if(doubled == replaced){
|
|
throw new InvalidCharacterException("The double letter replacement cannot be the same as the letter replaced");
|
|
}
|
|
|
|
if(!preserveCapitals){
|
|
logger.debug("Removing capitals");
|
|
|
|
doubled = Character.toUpperCase(doubled);
|
|
}
|
|
|
|
logger.debug("Setting doubled to {}", doubled);
|
|
this.doubled = doubled;
|
|
}
|
|
/**
|
|
* Sets the replacer character.
|
|
*
|
|
* @param replacer the character to replace the replaced character
|
|
* @throws InvalidCharacterException if the character is not a letter or is invalid
|
|
*/
|
|
protected void setReplacer(char replacer) throws InvalidCharacterException{
|
|
logger.debug("Setting replacer");
|
|
logger.debug("Original character {}", replacer);
|
|
|
|
//Make sure the character is a letter
|
|
logger.debug("Checking letter");
|
|
if(!Character.isAlphabetic(replacer)){
|
|
throw new InvalidCharacterException("The replacer must be a letter");
|
|
}
|
|
|
|
//Make sure the character isn't the same as what it is supposed to replace
|
|
logger.debug("Checking same as replaced");
|
|
if(replacer == replaced){
|
|
throw new InvalidCharacterException("The replacer cannot be the same letter as what it is replacing");
|
|
}
|
|
|
|
//Make sure the replacer isn't the same as the double letter replacer
|
|
logger.debug("Checking same as doubled");
|
|
if(replacer == doubled){
|
|
throw new InvalidCharacterException("The replacer cannot be the same as the double letter replacer");
|
|
}
|
|
|
|
if(!preserveCapitals){
|
|
logger.debug("Removing capitals");
|
|
|
|
replacer = Character.toUpperCase(replacer);
|
|
}
|
|
|
|
logger.debug("Setting replacer to {}", replacer);
|
|
this.replacer = replacer;
|
|
}
|
|
/**
|
|
* Sets the replaced character.
|
|
*
|
|
* @param replaced the character to be replaced
|
|
* @throws InvalidCharacterException if the character is not a letter or is invalid
|
|
*/
|
|
protected void setReplaced(char replaced) throws InvalidCharacterException{
|
|
logger.debug("Setting replaced");
|
|
logger.debug("Original character {}", replaced);
|
|
|
|
//Make sure the character is a letter
|
|
logger.debug("Checking letter");
|
|
if(!Character.isAlphabetic(replaced)){
|
|
throw new InvalidCharacterException("The replaced character must be a letter");
|
|
}
|
|
|
|
//Make sure the character isn't the same as what is replacing it
|
|
logger.debug("Checking same as replacer");
|
|
if(replaced == replacer){
|
|
throw new InvalidCharacterException("The replaced letter cannot be the same as the replacing letter");
|
|
}
|
|
|
|
//Make sure the replacer isn't the same as the double letter replacer
|
|
logger.debug("Checking same as doubled");
|
|
if(replaced == doubled){
|
|
throw new InvalidCharacterException("The replaced letter cannot be the same as the replacing letter");
|
|
}
|
|
|
|
if(!preserveCapitals){
|
|
logger.debug("Removing capitals");
|
|
|
|
replaced = Character.toUpperCase(replaced);
|
|
}
|
|
|
|
logger.debug("Setting replaced to {}", replaced);
|
|
this.replaced = replaced;
|
|
}
|
|
/**
|
|
* Creates the grid from the keyword.
|
|
*/
|
|
protected void createGrid(){
|
|
logger.debug("Creating grid from keyword");
|
|
|
|
for(int row = 0;row < 5;++row){
|
|
for(int col = 0;col < 5;++col){
|
|
char letter = keyword.charAt((5 * row) + col);
|
|
logger.debug("Letter {} going to position [{}] [{}]", letter, row, col);
|
|
grid[row][col] = letter;
|
|
}
|
|
}
|
|
|
|
logger.debug("Grid\n{}", getGrid());
|
|
}
|
|
/**
|
|
* Strips invalid characters from the string that needs encoding/decoding.
|
|
*
|
|
* @param inputString the input string to be cleaned
|
|
* @param encoding true if encoding, false if decoding
|
|
* @throws InvalidCharacterException if an invalid character is found
|
|
* @throws InvalidInputException if the input string is invalid
|
|
*/
|
|
protected void setInputString(String inputString, boolean encoding) throws InvalidCharacterException, InvalidInputException{
|
|
logger.debug("Setting input string");
|
|
|
|
//Make sure the input string is not null
|
|
if(inputString == null){
|
|
throw new InvalidInputException("The input string cannot be null");
|
|
}
|
|
|
|
logger.debug("Original input string {}", inputString);
|
|
|
|
//Set the options
|
|
if(!preserveCapitals){
|
|
logger.debug("Removing case");
|
|
|
|
inputString = inputString.toUpperCase();
|
|
}
|
|
if(!preserveWhitespace){
|
|
logger.debug("Removing whitespace");
|
|
|
|
inputString = inputString.replaceAll("\\s", "");
|
|
}
|
|
if(!preserveSymbols){
|
|
logger.debug("Removing symbols");
|
|
|
|
inputString = inputString.replaceAll("[^a-zA-Z\\s]", "");
|
|
}
|
|
|
|
//If there is nothing in the input string throw an exception
|
|
if(inputString.isBlank()){
|
|
throw new InvalidInputException("The input string cannot be blank");
|
|
}
|
|
|
|
//If this is encoding parse it and clean up any problems
|
|
if(encoding){
|
|
setInputStringEncode(inputString);
|
|
}
|
|
//If this is decoding just add it without parsing it
|
|
else{
|
|
//Throw an exception if replaced is included
|
|
if(inputString.contains(Character.toString(replaced))){
|
|
throw new InvalidCharacterException("An encoded message cannot contain a letter that needs replaced");
|
|
}
|
|
|
|
logger.debug("Clean input string '{}'", inputString);
|
|
this.inputString = inputString;
|
|
|
|
if((getPreparedInputString().length() % 2) == 1){
|
|
throw new InvalidInputException("Input length must be even");
|
|
}
|
|
}
|
|
|
|
if(getPreparedInputString().isBlank()){
|
|
throw new InvalidInputException("Input must have at least 1 letter");
|
|
}
|
|
}
|
|
/**
|
|
* Cleans up the input string for encoding.
|
|
*
|
|
* @param inputString the input string to be cleaned
|
|
*/
|
|
protected void setInputStringEncode(String inputString){
|
|
logger.debug("Cleaning up input string for encoding");
|
|
|
|
//Replace characters that need replaced
|
|
logger.debug("Replacing all {} with {}", replaced, replacer);
|
|
inputString = inputString.replace(Character.toString(replaced), Character.toString(replacer));
|
|
|
|
//Check if there are any doubled characters
|
|
StringBuilder cleanInput = new StringBuilder();
|
|
int letterCount = 0;
|
|
for(int cnt = 0;cnt < inputString.length();){
|
|
logger.debug("Starting at character {}", cnt);
|
|
//Advance until you find a letter, saving the input for inclusion
|
|
StringBuilder prepend = new StringBuilder();
|
|
while((cnt < inputString.length()) && !Character.isAlphabetic(inputString.charAt(cnt))){
|
|
prepend.append(inputString.charAt(cnt++));
|
|
}
|
|
//If we have reached the end of the string end the loop
|
|
if(cnt == inputString.length()){
|
|
cleanInput.append(prepend);
|
|
break;
|
|
}
|
|
//Get the next character
|
|
char firstLetter = inputString.charAt(cnt++);
|
|
++letterCount;
|
|
|
|
//Advance until you find a letter, saving the input for inclusion
|
|
StringBuilder middle = new StringBuilder();
|
|
while((cnt < inputString.length()) && !Character.isAlphabetic(inputString.charAt(cnt))){
|
|
middle.append(inputString.charAt(cnt++));
|
|
}
|
|
char secondLetter = '\0';
|
|
//If we have not reached the end of the string get the next character
|
|
if(cnt != inputString.length()){
|
|
secondLetter = inputString.charAt(cnt++);
|
|
++letterCount;
|
|
}
|
|
//If the second character is the same as the first character set the pointer back and use the doubled character
|
|
if(secondLetter == firstLetter){
|
|
--cnt;
|
|
secondLetter = doubled;
|
|
}
|
|
|
|
//Add all of the gathered input to the cleaned up input
|
|
logger.debug("Adding to clean input: {} {} {} {}", prepend, firstLetter, middle, secondLetter);
|
|
cleanInput.append(prepend);
|
|
cleanInput.append(firstLetter);
|
|
cleanInput.append(middle);
|
|
if(secondLetter != '\0'){
|
|
cleanInput.append(secondLetter);
|
|
}
|
|
}
|
|
|
|
//Check if there are an odd number of characters
|
|
logger.debug("Checking odd characters");
|
|
if((letterCount % 2) == 1){
|
|
logger.debug("Adding final character to make even");
|
|
int lastLetterLocation = cleanInput.length() - 1;
|
|
while(!Character.isAlphabetic(cleanInput.charAt(lastLetterLocation))){
|
|
--lastLetterLocation;
|
|
}
|
|
if(cleanInput.charAt(lastLetterLocation) == doubled){
|
|
cleanInput.append(replacer);
|
|
}
|
|
else{
|
|
cleanInput.append(doubled);
|
|
}
|
|
}
|
|
|
|
this.inputString = cleanInput.toString();
|
|
logger.debug("Cleaned input string '{}'", this.inputString);
|
|
}
|
|
/**
|
|
* Returns the input string ready for encoding.
|
|
*
|
|
* @return the prepared input string
|
|
*/
|
|
protected String getPreparedInputString(){
|
|
logger.debug("Getting input string ready for encoding");
|
|
|
|
String cleanString = inputString.toUpperCase();
|
|
cleanString = cleanString.replaceAll("[^A-Z]", "");
|
|
|
|
logger.debug("Prepared string '{}'", cleanString);
|
|
|
|
return cleanString;
|
|
}
|
|
/**
|
|
* Strips invalid characters from the keyword and creates the grid.
|
|
*
|
|
* @param keyword the keyword to be processed
|
|
* @throws InvalidKeywordException if the keyword is invalid
|
|
*/
|
|
protected void setKeyword(String keyword) throws InvalidKeywordException{
|
|
logger.debug("Setting keyword");
|
|
|
|
if(keyword == null){
|
|
throw new InvalidKeywordException("Keyword cannot be null");
|
|
}
|
|
|
|
logger.debug("Original keyword {}", keyword);
|
|
|
|
//Change everything to uppercase
|
|
logger.debug("Removing case");
|
|
keyword = keyword.toUpperCase();
|
|
|
|
//Removing everything except capital letters
|
|
logger.debug("Removing all non-letter characters");
|
|
keyword = keyword.replaceAll("[^A-Z]", "");
|
|
|
|
//Add all letters in the alphabet to the key
|
|
logger.debug("Appending the alphabet to the keyword");
|
|
keyword += "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
|
|
|
//Replace all replaced characters
|
|
logger.debug("Replacing {} with {}", replaced, replacer);
|
|
keyword = keyword.replaceAll(Character.toString(Character.toUpperCase(replaced)), Character.toString(Character.toUpperCase(replacer)));
|
|
|
|
//Remove all duplicate chatacters
|
|
logger.debug("Removing duplicate characters");
|
|
StringBuilder uniqueKey = new StringBuilder();
|
|
keyword.chars().distinct().forEach(c -> uniqueKey.append((char)c));
|
|
this.keyword = uniqueKey.toString();
|
|
logger.debug("Cleaned keyword {}", this.keyword);
|
|
|
|
//Create the grid from the sanitized keyword
|
|
createGrid();
|
|
}
|
|
/**
|
|
* Returns the location of the given character in the grid.
|
|
*
|
|
* @param letter the character whose location is to be found
|
|
* @return the location of the character in the grid
|
|
* @throws InvalidInputException if the character is not found in the grid
|
|
*/
|
|
protected CharLocation findChar(char letter) throws InvalidInputException{
|
|
logger.debug("Finding character in grid {}", letter);
|
|
|
|
for(int row = 0;row < grid.length;++row){
|
|
for(int col = 0;col < grid[row].length;++col){
|
|
if(grid[row][col] == letter){
|
|
logger.debug("Found at {}, {}", row, col);
|
|
|
|
return new CharLocation(row, col);
|
|
}
|
|
}
|
|
}
|
|
|
|
//If it was not found something went wrong
|
|
throw new InvalidInputException("The character '" + letter + "' was not found in the grid");
|
|
}
|
|
/**
|
|
* Returns the character in the grid at the specified coordinates.
|
|
*
|
|
* @param x the x-coordinate
|
|
* @param y the y-coordinate
|
|
* @return the character at the specified coordinates
|
|
*/
|
|
protected char getGridChar(int x, int y){
|
|
logger.debug("Getting character from grid[{}][{}]", x, y);
|
|
|
|
if(x < 0){
|
|
x += 5;
|
|
}
|
|
if(y < 0){
|
|
y += 5;
|
|
}
|
|
|
|
char letter = grid[x % 5][y % 5];
|
|
logger.debug("Character {}", letter);
|
|
return letter;
|
|
}
|
|
/**
|
|
* Adds characters that aren't letters to the output.
|
|
*
|
|
* @param cleanString the cleaned string to be formatted
|
|
*/
|
|
protected void addCharactersToCleanString(String cleanString){
|
|
logger.debug("Formatting output string");
|
|
|
|
int outputCnt = 0;
|
|
StringBuilder fullOutput = new StringBuilder();
|
|
for(int inputCnt = 0;inputCnt < inputString.length();++inputCnt){
|
|
logger.debug("Working character {}", inputString.charAt(inputCnt));
|
|
|
|
if(Character.isUpperCase(inputString.charAt(inputCnt))){
|
|
logger.debug("Appending uppercase");
|
|
|
|
fullOutput.append(cleanString.charAt(outputCnt++));
|
|
}
|
|
else if(Character.isLowerCase(inputString.charAt(inputCnt))){
|
|
logger.debug("Appending lowercase");
|
|
|
|
fullOutput.append(Character.toLowerCase(cleanString.charAt(outputCnt++)));
|
|
}
|
|
else{
|
|
logger.debug("Appending symbol");
|
|
|
|
fullOutput.append(inputString.charAt(inputCnt));
|
|
}
|
|
}
|
|
|
|
outputString = fullOutput.toString();
|
|
logger.debug("Formatted output '{}'", outputString);
|
|
}
|
|
/**
|
|
* Encodes the input string using the Playfair cipher and stores the result in the output string.
|
|
*
|
|
* @throws InvalidInputException if the input string is invalid
|
|
*/
|
|
protected void encode() throws InvalidInputException{
|
|
logger.debug("Encoding");
|
|
|
|
StringBuilder output = new StringBuilder();
|
|
int inputCnt = 0;
|
|
String cleanString = getPreparedInputString();
|
|
while(inputCnt < cleanString.length()){
|
|
//Get the next 2 letters to be encoded
|
|
char firstLetter = cleanString.charAt(inputCnt++);
|
|
char secondLetter = cleanString.charAt(inputCnt++);
|
|
|
|
logger.debug("Letters {} {}", firstLetter, secondLetter);
|
|
|
|
//Find the letters in the grid
|
|
CharLocation firstLocation = findChar(firstLetter);
|
|
CharLocation secondLocation = findChar(secondLetter);
|
|
|
|
//Encode the letters
|
|
if(firstLocation.getX() == secondLocation.getX()){
|
|
logger.debug("Row encoding");
|
|
firstLetter = getGridChar(firstLocation.getX(), firstLocation.getY() + 1);
|
|
secondLetter = getGridChar(secondLocation.getX(), secondLocation.getY() + 1);
|
|
}
|
|
else if(firstLocation.getY() == secondLocation.getY()){
|
|
logger.debug("Column encoding");
|
|
firstLetter = getGridChar(firstLocation.getX() + 1, firstLocation.getY());
|
|
secondLetter = getGridChar(secondLocation.getX() + 1, secondLocation.getY());
|
|
}
|
|
else{
|
|
logger.debug("Corner encoding");
|
|
firstLetter = getGridChar(firstLocation.getX(), secondLocation.getY());
|
|
secondLetter = getGridChar(secondLocation.getX(), firstLocation.getY());
|
|
}
|
|
|
|
//Add the new letters to the output string
|
|
logger.debug("Encoded letters {} {}", firstLetter, secondLetter);
|
|
output.append(firstLetter);
|
|
output.append(secondLetter);
|
|
}
|
|
|
|
//Add other characters to the output string
|
|
addCharactersToCleanString(output.toString());
|
|
}
|
|
/**
|
|
* Decodes the input string using the Playfair cipher and stores the result in the output string.
|
|
*
|
|
* @throws InvalidInputException if the input string is invalid
|
|
*/
|
|
protected void decode() throws InvalidInputException{
|
|
logger.debug("Decoding");
|
|
|
|
StringBuilder output = new StringBuilder();
|
|
int inputCnt = 0;
|
|
String cleanString = getPreparedInputString();
|
|
while(inputCnt < cleanString.length()){
|
|
//Get the next 2 letters to be encoded
|
|
char firstLetter = cleanString.charAt(inputCnt++);
|
|
char secondLetter = cleanString.charAt(inputCnt++);
|
|
|
|
logger.debug("Letters {} {}", firstLetter, secondLetter);
|
|
|
|
//Find the letters in the grid
|
|
CharLocation firstLocation = findChar(firstLetter);
|
|
CharLocation secondLocation = findChar(secondLetter);
|
|
|
|
//Decode the letters
|
|
if(firstLocation.getX() == secondLocation.getX()){
|
|
logger.debug("Row decoding");
|
|
firstLetter = getGridChar(firstLocation.getX(), firstLocation.getY() - 1);
|
|
secondLetter = getGridChar(secondLocation.getX(), secondLocation.getY() - 1);
|
|
}
|
|
else if(firstLocation.getY() == secondLocation.getY()){
|
|
logger.debug("Column decoding");
|
|
firstLetter = getGridChar(firstLocation.getX() - 1, firstLocation.getY());
|
|
secondLetter = getGridChar(secondLocation.getX() - 1, secondLocation.getY());
|
|
}
|
|
else{
|
|
logger.debug("Corner decoding");
|
|
firstLetter = getGridChar(firstLocation.getX(), secondLocation.getY());
|
|
secondLetter = getGridChar(secondLocation.getX(), firstLocation.getY());
|
|
}
|
|
|
|
//Add the new letters to the output string
|
|
logger.debug("Decoded letters {} {}", firstLetter, secondLetter);
|
|
output.append(firstLetter);
|
|
output.append(secondLetter);
|
|
}
|
|
|
|
//Add other characters to the output string
|
|
addCharactersToCleanString(output.toString());
|
|
}
|
|
|
|
//?Constructor
|
|
/**
|
|
* Constructs a Playfair cipher instance with default settings.
|
|
*
|
|
* @throws InvalidCharacterException if default characters are invalid
|
|
*/
|
|
public Playfair() throws InvalidCharacterException{
|
|
reset();
|
|
preserveCapitals = false;
|
|
preserveWhitespace = false;
|
|
preserveSymbols = false;
|
|
setReplaced('j');
|
|
setReplacer('i');
|
|
setDoubled('x');
|
|
}
|
|
/**
|
|
* Constructs a Playfair cipher instance with specified settings.
|
|
*
|
|
* @param preserveCapitals whether to preserve capital letters
|
|
* @param preserveWhitespace whether to preserve whitespace
|
|
* @param preserveSymbols whether to preserve symbols
|
|
* @throws InvalidCharacterException if default characters are invalid
|
|
*/
|
|
public Playfair(boolean preserveCapitals, boolean preserveWhitespace, boolean preserveSymbols) throws InvalidCharacterException{
|
|
reset();
|
|
this.preserveCapitals = preserveCapitals;
|
|
this.preserveWhitespace = preserveWhitespace;
|
|
this.preserveSymbols = preserveSymbols;
|
|
setReplaced('j');
|
|
setReplacer('i');
|
|
setDoubled('x');
|
|
}
|
|
/**
|
|
* Constructs a Playfair cipher instance with specified settings and characters.
|
|
*
|
|
* @param preserveCapitals whether to preserve capital letters
|
|
* @param preserveWhitespace whether to preserve whitespace
|
|
* @param preserveSymbols whether to preserve symbols
|
|
* @param replaced the character to be replaced
|
|
* @param replacer the character to replace the replaced character
|
|
* @param doubled the character to use for doubled letters
|
|
* @throws InvalidCharacterException if any character is invalid
|
|
*/
|
|
public Playfair(boolean preserveCapitals, boolean preserveWhitespace, boolean preserveSymbols, char replaced, char replacer, char doubled) throws InvalidCharacterException{
|
|
reset();
|
|
this.preserveCapitals = preserveCapitals;
|
|
this.preserveWhitespace = preserveWhitespace;
|
|
this.preserveSymbols = preserveSymbols;
|
|
setReplaced(replaced);
|
|
setReplacer(replacer);
|
|
setDoubled(doubled);
|
|
}
|
|
|
|
/**
|
|
* Sets the keyword and input string and encodes the message.
|
|
*
|
|
* @param keyword the keyword for the cipher
|
|
* @param input the message to encode
|
|
* @return the encoded message
|
|
* @throws InvalidCharacterException if any character is invalid
|
|
* @throws InvalidInputException if the input string is invalid
|
|
*/
|
|
public String encode(String keyword, String input) throws InvalidCharacterException, InvalidInputException{
|
|
reset();
|
|
setKeyword(keyword);
|
|
setInputString(input, true);
|
|
encode();
|
|
return outputString;
|
|
}
|
|
/**
|
|
* Sets the keyword and input string and decodes the message.
|
|
*
|
|
* @param keyword the keyword for the cipher
|
|
* @param input the encoded message to decode
|
|
* @return the decoded message
|
|
* @throws InvalidCharacterException if any character is invalid
|
|
* @throws InvalidInputException if the input string is invalid
|
|
*/
|
|
public String decode(String keyword, String input) throws InvalidCharacterException, InvalidInputException{
|
|
reset();
|
|
setKeyword(keyword);
|
|
setInputString(input, false);
|
|
decode();
|
|
return outputString;
|
|
}
|
|
|
|
/**
|
|
* Resets all variables to their default values.
|
|
*/
|
|
public void reset(){
|
|
logger.debug("Resetting fields");
|
|
|
|
grid = new char[5][5];
|
|
inputString = "";
|
|
outputString = "";
|
|
keyword = "";
|
|
}
|
|
//?Getters
|
|
/**
|
|
* Returns the replaced character.
|
|
*
|
|
* @return the replaced character
|
|
*/
|
|
public char getReplaced(){
|
|
return replaced;
|
|
}
|
|
/**
|
|
* Returns the replacer character.
|
|
*
|
|
* @return the replacer character
|
|
*/
|
|
public char getReplacer(){
|
|
return replacer;
|
|
}
|
|
/**
|
|
* Returns the doubled character.
|
|
*
|
|
* @return the doubled character
|
|
*/
|
|
public char getDoubled(){
|
|
return doubled;
|
|
}
|
|
/**
|
|
* Returns the keyword used in the cipher.
|
|
*
|
|
* @return the keyword
|
|
*/
|
|
public String getKeyword(){
|
|
return keyword;
|
|
}
|
|
/**
|
|
* Returns the input string that was set for encoding/decoding.
|
|
*
|
|
* @return the input string
|
|
*/
|
|
public String getInputString(){
|
|
return inputString;
|
|
}
|
|
/**
|
|
* Returns the output string after encoding/decoding.
|
|
*
|
|
* @return the output string
|
|
*/
|
|
public String getOutputString(){
|
|
return outputString;
|
|
}
|
|
/**
|
|
* Return the grid used for encoding/decoding.
|
|
*
|
|
* @return the grid as a string
|
|
*/
|
|
public String getGrid(){
|
|
logger.debug("Creating string from grid");
|
|
|
|
StringJoiner gridString = new StringJoiner("\n");
|
|
for(char[] row : grid){
|
|
StringJoiner rowString = new StringJoiner(" ", "[", "]");
|
|
for(char col : row){
|
|
rowString.add(Character.toString(col));
|
|
}
|
|
gridString.add(rowString.toString());
|
|
}
|
|
|
|
return gridString.toString();
|
|
}
|
|
}
|