Enhance docs, add more tests in Vigenere (#5899)

This commit is contained in:
Hardik Pawar
2024-10-27 00:20:31 +05:30
committed by GitHub
parent e6f70634a4
commit 8b604858f5
2 changed files with 118 additions and 19 deletions

View File

@ -1,16 +1,54 @@
package com.thealgorithms.ciphers; package com.thealgorithms.ciphers;
/** /**
* A Java implementation of Vigenere Cipher. * A Java implementation of the Vigenère Cipher.
*
* The Vigenère Cipher is a polyalphabetic substitution cipher that uses a
* keyword to shift letters in the plaintext by different amounts, depending
* on the corresponding character in the keyword. It wraps around the alphabet,
* ensuring the shifts are within 'A'-'Z' or 'a'-'z'.
*
* Non-alphabetic characters (like spaces, punctuation) are kept unchanged.
*
* Encryption Example:
* - Plaintext: "Hello World!"
* - Key: "suchsecret"
* - Encrypted Text: "Zynsg Yfvev!"
*
* Decryption Example:
* - Ciphertext: "Zynsg Yfvev!"
* - Key: "suchsecret"
* - Decrypted Text: "Hello World!"
*
* Wikipedia Reference:
* <a href="https://en.wikipedia.org/wiki/Vigen%C3%A8re_cipher">Vigenère Cipher - Wikipedia</a>
* *
* @author straiffix * @author straiffix
* @author beingmartinbmc * @author beingmartinbmc
*/ */
public class Vigenere { public class Vigenere {
/**
* Encrypts a given message using the Vigenère Cipher with the specified key.
* Steps:
* 1. Iterate over each character in the message.
* 2. If the character is a letter, shift it by the corresponding character in the key.
* 3. Preserve the case of the letter.
* 4. Preserve non-alphabetic characters.
* 5. Move to the next character in the key (cyclic).
* 6. Return the encrypted message.
*
* @param message The plaintext message to encrypt.
* @param key The keyword used for encryption.
* @throws IllegalArgumentException if the key is empty.
* @return The encrypted message.
*/
public String encrypt(final String message, final String key) { public String encrypt(final String message, final String key) {
StringBuilder result = new StringBuilder(); if (key.isEmpty()) {
throw new IllegalArgumentException("Key cannot be empty.");
}
StringBuilder result = new StringBuilder();
int j = 0; int j = 0;
for (int i = 0; i < message.length(); i++) { for (int i = 0; i < message.length(); i++) {
char c = message.charAt(i); char c = message.charAt(i);
@ -20,17 +58,35 @@ public class Vigenere {
} else { } else {
result.append((char) ((c + key.toLowerCase().charAt(j) - 2 * 'a') % 26 + 'a')); result.append((char) ((c + key.toLowerCase().charAt(j) - 2 * 'a') % 26 + 'a'));
} }
j = ++j % key.length();
} else { } else {
result.append(c); result.append(c);
} }
j = ++j % key.length();
} }
return result.toString(); return result.toString();
} }
/**
* Decrypts a given message encrypted with the Vigenère Cipher using the specified key.
* Steps:
* 1. Iterate over each character in the message.
* 2. If the character is a letter, shift it back by the corresponding character in the key.
* 3. Preserve the case of the letter.
* 4. Preserve non-alphabetic characters.
* 5. Move to the next character in the key (cyclic).
* 6. Return the decrypted message.
*
* @param message The encrypted message to decrypt.
* @param key The keyword used for decryption.
* @throws IllegalArgumentException if the key is empty.
* @return The decrypted plaintext message.
*/
public String decrypt(final String message, final String key) { public String decrypt(final String message, final String key) {
StringBuilder result = new StringBuilder(); if (key.isEmpty()) {
throw new IllegalArgumentException("Key cannot be empty.");
}
StringBuilder result = new StringBuilder();
int j = 0; int j = 0;
for (int i = 0; i < message.length(); i++) { for (int i = 0; i < message.length(); i++) {
char c = message.charAt(i); char c = message.charAt(i);
@ -40,11 +96,10 @@ public class Vigenere {
} else { } else {
result.append((char) ('z' - (25 - (c - key.toLowerCase().charAt(j))) % 26)); result.append((char) ('z' - (25 - (c - key.toLowerCase().charAt(j))) % 26));
} }
j = ++j % key.length();
} else { } else {
result.append(c); result.append(c);
} }
j = ++j % key.length();
} }
return result.toString(); return result.toString();
} }

View File

@ -1,6 +1,7 @@
package com.thealgorithms.ciphers; package com.thealgorithms.ciphers;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
@ -9,28 +10,71 @@ class VigenereTest {
Vigenere vigenere = new Vigenere(); Vigenere vigenere = new Vigenere();
@Test @Test
void vigenereEncryptTest() { void testVigenereEncryptDecrypt() {
// given
String text = "Hello World!"; String text = "Hello World!";
String key = "suchsecret"; String key = "suchsecret";
// when String encryptedText = vigenere.encrypt(text, key);
String cipherText = vigenere.encrypt(text, key); String decryptedText = vigenere.decrypt(encryptedText, key);
// then assertEquals("Zynsg Aqipw!", encryptedText);
assertEquals("Zynsg Yfvev!", cipherText); assertEquals("Hello World!", decryptedText);
} }
@Test @Test
void vigenereDecryptTest() { void testWithEmptyMessage() {
// given String text = "";
String encryptedText = "Zynsg Yfvev!"; String key = "anykey";
String key = "suchsecret";
// when String encryptedText = vigenere.encrypt(text, key);
String decryptedText = vigenere.decrypt(encryptedText, key); String decryptedText = vigenere.decrypt(encryptedText, key);
// then assertEquals("", encryptedText);
assertEquals("Hello World!", decryptedText); assertEquals("", decryptedText);
}
@Test
void testWithEmptyKey() {
String text = "This should remain the same";
String key = "";
assertThrows(IllegalArgumentException.class, () -> vigenere.encrypt(text, key));
assertThrows(IllegalArgumentException.class, () -> vigenere.decrypt(text, key));
}
@Test
void testWithNumbersInMessage() {
String text = "Vigenere123!";
String key = "cipher";
String encryptedText = vigenere.encrypt(text, key);
String decryptedText = vigenere.decrypt(encryptedText, key);
assertEquals("Xqvlrvtm123!", encryptedText);
assertEquals(text, decryptedText);
}
@Test
void testLongerKeyThanMessage() {
String text = "Short";
String key = "VeryLongSecretKey";
String encryptedText = vigenere.encrypt(text, key);
String decryptedText = vigenere.decrypt(encryptedText, key);
assertEquals("Nlfpe", encryptedText);
assertEquals(text, decryptedText);
}
@Test
void testUppercaseMessageAndKey() {
String text = "HELLO";
String key = "SECRET";
String encryptedText = vigenere.encrypt(text, key);
String decryptedText = vigenere.decrypt(encryptedText, key);
assertEquals("ZINCS", encryptedText);
assertEquals(text, decryptedText);
} }
} }