NIFI-8319 Added AES/CBC/NoPadding for decryption in EncryptContent Processor

Signed-off-by: Pierre Villard <pierre.villard.fr@gmail.com>

This closes #4894.
This commit is contained in:
exceptionfactory 2021-03-12 16:00:42 -06:00 committed by Pierre Villard
parent 8202bffc98
commit a7ba5bb4f8
No known key found for this signature in database
GPG Key ID: F92A93B30C07C6D5
9 changed files with 113 additions and 128 deletions

View File

@ -49,7 +49,8 @@ public enum EncryptionMethod {
SHA_TWOFISH("PBEWITHSHAANDTWOFISH-CBC", "BC", false, false), SHA_TWOFISH("PBEWITHSHAANDTWOFISH-CBC", "BC", false, false),
PGP("PGP", "BC", false, false), PGP("PGP", "BC", false, false),
PGP_ASCII_ARMOR("PGP-ASCII-ARMOR", "BC", false, false), PGP_ASCII_ARMOR("PGP-ASCII-ARMOR", "BC", false, false),
// New encryption methods which used keyed encryption // AES/CBC/NoPadding supported for decryption
AES_CBC_NO_PADDING("AES/CBC/NoPadding", "BC", false, true),
AES_CBC("AES/CBC/PKCS7Padding", "BC", false, true), AES_CBC("AES/CBC/PKCS7Padding", "BC", false, true),
AES_CTR("AES/CTR/NoPadding", "BC", false, true), AES_CTR("AES/CTR/NoPadding", "BC", false, true),
AES_GCM("AES/GCM/NoPadding", "BC", false, true); AES_GCM("AES/GCM/NoPadding", "BC", false, true);

View File

@ -43,6 +43,8 @@ class AESKeyedCipherProviderGroovyTest {
private static final String KEY_HEX = "0123456789ABCDEFFEDCBA9876543210" private static final String KEY_HEX = "0123456789ABCDEFFEDCBA9876543210"
private static final String PLAINTEXT = "ExactBlockSizeRequiredForProcess"
private static final List<EncryptionMethod> keyedEncryptionMethods = EncryptionMethod.values().findAll { it.keyedCipher } private static final List<EncryptionMethod> keyedEncryptionMethods = EncryptionMethod.values().findAll { it.keyedCipher }
private static final SecretKey key = new SecretKeySpec(Hex.decodeHex(KEY_HEX as char[]), "AES") private static final SecretKey key = new SecretKeySpec(Hex.decodeHex(KEY_HEX as char[]), "AES")
@ -73,8 +75,6 @@ class AESKeyedCipherProviderGroovyTest {
// Arrange // Arrange
KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider() KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider()
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : keyedEncryptionMethods) { for (EncryptionMethod em : keyedEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -84,7 +84,7 @@ class AESKeyedCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, key, iv, false) cipher = cipherProvider.getCipher(em, key, iv, false)
@ -93,7 +93,7 @@ class AESKeyedCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -102,8 +102,6 @@ class AESKeyedCipherProviderGroovyTest {
// Arrange // Arrange
KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider() KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider()
final String plaintext = "This is a plaintext message."
// Act // Act
keyedEncryptionMethods.each { EncryptionMethod em -> keyedEncryptionMethods.each { EncryptionMethod em ->
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -113,7 +111,7 @@ class AESKeyedCipherProviderGroovyTest {
// Initialize a cipher for encryption // Initialize a cipher for encryption
Cipher cipher = cipherProvider.getCipher(em, key, iv, true) Cipher cipher = cipherProvider.getCipher(em, key, iv, true)
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, key, iv, false) cipher = cipherProvider.getCipher(em, key, iv, false)
@ -122,7 +120,7 @@ class AESKeyedCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -134,8 +132,6 @@ class AESKeyedCipherProviderGroovyTest {
KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider() KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider()
final List<Integer> LONG_KEY_LENGTHS = [192, 256] final List<Integer> LONG_KEY_LENGTHS = [192, 256]
final String plaintext = "This is a plaintext message."
SecureRandom secureRandom = new SecureRandom() SecureRandom secureRandom = new SecureRandom()
// Act // Act
@ -156,7 +152,7 @@ class AESKeyedCipherProviderGroovyTest {
// Initialize a cipher for encryption // Initialize a cipher for encryption
Cipher cipher = cipherProvider.getCipher(em, localKey, iv, true) Cipher cipher = cipherProvider.getCipher(em, localKey, iv, true)
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, localKey, iv, false) cipher = cipherProvider.getCipher(em, localKey, iv, false)
@ -165,7 +161,7 @@ class AESKeyedCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
} }
@ -240,7 +236,7 @@ class AESKeyedCipherProviderGroovyTest {
// Arrange // Arrange
KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider() KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider()
final String PLAINTEXT = "This is a plaintext message." final String plaintext = "This is a plaintext message."
// These values can be generated by running `$ ./openssl_aes.rb` in the terminal // These values can be generated by running `$ ./openssl_aes.rb` in the terminal
final byte[] IV = Hex.decodeHex("e0bc8cc7fbc0bdfdc184dc22ce2fcb5b" as char[]) final byte[] IV = Hex.decodeHex("e0bc8cc7fbc0bdfdc184dc22ce2fcb5b" as char[])
@ -261,7 +257,7 @@ class AESKeyedCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert PLAINTEXT.equals(recovered) assert plaintext.equals(recovered)
} }
@Test @Test
@ -269,8 +265,6 @@ class AESKeyedCipherProviderGroovyTest {
// Arrange // Arrange
KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider() KeyedCipherProvider cipherProvider = new AESKeyedCipherProvider()
final String plaintext = "This is a plaintext message."
// Act // Act
keyedEncryptionMethods.each { EncryptionMethod em -> keyedEncryptionMethods.each { EncryptionMethod em ->
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -280,7 +274,7 @@ class AESKeyedCipherProviderGroovyTest {
// Initialize a cipher for encryption // Initialize a cipher for encryption
Cipher cipher = cipherProvider.getCipher(em, key, iv, true) Cipher cipher = cipherProvider.getCipher(em, key, iv, true)
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
def msg = shouldFail(IllegalArgumentException) { def msg = shouldFail(IllegalArgumentException) {

View File

@ -42,6 +42,8 @@ import static groovy.test.GroovyAssert.shouldFail
class Argon2CipherProviderGroovyTest extends GroovyTestCase { class Argon2CipherProviderGroovyTest extends GroovyTestCase {
private static final Logger logger = LoggerFactory.getLogger(Argon2CipherProviderGroovyTest.class) private static final Logger logger = LoggerFactory.getLogger(Argon2CipherProviderGroovyTest.class)
private static final String PLAINTEXT = "ExactBlockSizeRequiredForProcess"
private static List<EncryptionMethod> strongKDFEncryptionMethods private static List<EncryptionMethod> strongKDFEncryptionMethods
private static final int DEFAULT_KEY_LENGTH = 128 private static final int DEFAULT_KEY_LENGTH = 128
@ -85,8 +87,6 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
final String PASSWORD = "shortPassword" final String PASSWORD = "shortPassword"
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -96,7 +96,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false)
@ -105,7 +105,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -210,8 +210,6 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("01" * 16 as char[]) final byte[] IV = Hex.decodeHex("01" * 16 as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -220,7 +218,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false)
@ -229,7 +227,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -244,8 +242,6 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
final int LONG_KEY_LENGTH = 256 final int LONG_KEY_LENGTH = 256
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -255,7 +251,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false)
@ -264,7 +260,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -369,8 +365,6 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("00" * 16 as char[]) final byte[] IV = Hex.decodeHex("00" * 16 as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -379,7 +373,7 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
def msg = shouldFail(IllegalArgumentException) { def msg = shouldFail(IllegalArgumentException) {
@ -399,8 +393,6 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("01" * 16 as char[]) final byte[] IV = Hex.decodeHex("01" * 16 as char[])
final String PLAINTEXT = "This is a plaintext message."
final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
@ -432,8 +424,6 @@ class Argon2CipherProviderGroovyTest extends GroovyTestCase {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("00" * 16 as char[]) final byte[] IV = Hex.decodeHex("00" * 16 as char[])
final String PLAINTEXT = "This is a plaintext message."
final def INVALID_KEY_LENGTHS = [-1, 40, 64, 112, 512] final def INVALID_KEY_LENGTHS = [-1, 40, 64, 112, 512]
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC

View File

@ -46,6 +46,8 @@ import static org.junit.Assert.assertTrue
class BcryptCipherProviderGroovyTest { class BcryptCipherProviderGroovyTest {
private static final Logger logger = LoggerFactory.getLogger(BcryptCipherProviderGroovyTest.class) private static final Logger logger = LoggerFactory.getLogger(BcryptCipherProviderGroovyTest.class)
private static final String PLAINTEXT = "ExactBlockSizeRequiredForProcess"
private static List<EncryptionMethod> strongKDFEncryptionMethods private static List<EncryptionMethod> strongKDFEncryptionMethods
private static final int DEFAULT_KEY_LENGTH = 128 private static final int DEFAULT_KEY_LENGTH = 128
@ -86,8 +88,6 @@ class BcryptCipherProviderGroovyTest {
final String PASSWORD = "shortPassword" final String PASSWORD = "shortPassword"
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -97,7 +97,7 @@ class BcryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false)
@ -106,7 +106,7 @@ class BcryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -119,8 +119,6 @@ class BcryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("01" * 16 as char[]) final byte[] IV = Hex.decodeHex("01" * 16 as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -129,7 +127,7 @@ class BcryptCipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false)
@ -138,7 +136,7 @@ class BcryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -155,8 +153,6 @@ class BcryptCipherProviderGroovyTest {
final int LONG_KEY_LENGTH = 256 final int LONG_KEY_LENGTH = 256
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -166,7 +162,7 @@ class BcryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false)
@ -175,7 +171,7 @@ class BcryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -389,8 +385,6 @@ class BcryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("00" * 16 as char[]) final byte[] IV = Hex.decodeHex("00" * 16 as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -399,7 +393,7 @@ class BcryptCipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
def msg = shouldFail(IllegalArgumentException) { def msg = shouldFail(IllegalArgumentException) {
@ -420,8 +414,6 @@ class BcryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("01" * 16 as char[]) final byte[] IV = Hex.decodeHex("01" * 16 as char[])
final String PLAINTEXT = "This is a plaintext message."
// Currently only AES ciphers are compatible with Bcrypt, so redundant to test all algorithms // Currently only AES ciphers are compatible with Bcrypt, so redundant to test all algorithms
final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
@ -456,8 +448,6 @@ class BcryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("00" * 16 as char[]) final byte[] IV = Hex.decodeHex("00" * 16 as char[])
final String PLAINTEXT = "This is a plaintext message."
// Currently only AES ciphers are compatible with Bcrypt, so redundant to test all algorithms // Currently only AES ciphers are compatible with Bcrypt, so redundant to test all algorithms
final def INVALID_KEY_LENGTHS = [-1, 40, 64, 112, 512] final def INVALID_KEY_LENGTHS = [-1, 40, 64, 112, 512]
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
@ -520,8 +510,6 @@ class BcryptCipherProviderGroovyTest {
byte[] keyDigestBytes = sha512.digest(hashOutputBytes[-31..-1] as byte[]) byte[] keyDigestBytes = sha512.digest(hashOutputBytes[-31..-1] as byte[])
logger.info("Key digest (${keyDigestBytes.length}): ${Hex.encodeHexString(keyDigestBytes)}") logger.info("Key digest (${keyDigestBytes.length}): ${Hex.encodeHexString(keyDigestBytes)}")
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -531,7 +519,7 @@ class BcryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false)
@ -551,8 +539,8 @@ class BcryptCipherProviderGroovyTest {
logger.info("Verified: ${verificationRecovered}") logger.info("Verified: ${verificationRecovered}")
// Assert // Assert
assert plaintext == recovered assert PLAINTEXT == recovered
assert plaintext == verificationRecovered assert PLAINTEXT == verificationRecovered
} }
} }
@ -564,8 +552,6 @@ class BcryptCipherProviderGroovyTest {
final String PASSWORD = "shortPassword" final String PASSWORD = "shortPassword"
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -575,7 +561,7 @@ class BcryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getLegacyDecryptCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH) cipher = cipherProvider.getLegacyDecryptCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH)
@ -584,7 +570,7 @@ class BcryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext == recovered assert PLAINTEXT == recovered
} }
} }
@ -597,8 +583,6 @@ class BcryptCipherProviderGroovyTest {
final byte[] SALT = null final byte[] SALT = null
final EncryptionMethod em = EncryptionMethod.AES_CBC final EncryptionMethod em = EncryptionMethod.AES_CBC
final String plaintext = "This is a plaintext message."
// Act // Act
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -610,7 +594,7 @@ class BcryptCipherProviderGroovyTest {
} }
logger.expected("Encrypt error: ${encryptMsg}") logger.expected("Encrypt error: ${encryptMsg}")
byte[] cipherBytes = plaintext.reverse().getBytes(StandardCharsets.UTF_8) byte[] cipherBytes = PLAINTEXT.reverse().getBytes(StandardCharsets.UTF_8)
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
def decryptMsg = shouldFail(IllegalArgumentException) { def decryptMsg = shouldFail(IllegalArgumentException) {

View File

@ -40,7 +40,7 @@ class CipherUtilityGroovyTest extends GroovyTestCase {
private static final List<String> SYMMETRIC_ALGORITHMS = EncryptionMethod.values().findAll { it.algorithm.startsWith("PBE") || it.algorithm.startsWith("AES") }*.algorithm private static final List<String> SYMMETRIC_ALGORITHMS = EncryptionMethod.values().findAll { it.algorithm.startsWith("PBE") || it.algorithm.startsWith("AES") }*.algorithm
private static final Map<String, List<String>> ALGORITHMS_MAPPED_BY_CIPHER = SYMMETRIC_ALGORITHMS.groupBy { String algorithm -> CIPHERS.find { algorithm.contains(it) } } private static final Map<String, List<String>> ALGORITHMS_MAPPED_BY_CIPHER = SYMMETRIC_ALGORITHMS.groupBy { String algorithm -> CIPHERS.find { algorithm.contains(it) } }
// Manually mapped as of 01/19/16 0.5.0 // Manually mapped as of 03/21/21 1.13.0
private static final Map<Integer, List<String>> ALGORITHMS_MAPPED_BY_KEY_LENGTH = [ private static final Map<Integer, List<String>> ALGORITHMS_MAPPED_BY_KEY_LENGTH = [
(40) : ["PBEWITHSHAAND40BITRC2-CBC", (40) : ["PBEWITHSHAAND40BITRC2-CBC",
"PBEWITHSHAAND40BITRC4"], "PBEWITHSHAAND40BITRC4"],
@ -56,18 +56,21 @@ class CipherUtilityGroovyTest extends GroovyTestCase {
"PBEWITHSHAAND128BITRC2-CBC", "PBEWITHSHAAND128BITRC2-CBC",
"PBEWITHSHAAND128BITRC4", "PBEWITHSHAAND128BITRC4",
"PBEWITHSHAANDTWOFISH-CBC", "PBEWITHSHAANDTWOFISH-CBC",
"AES/CBC/NoPadding",
"AES/CBC/PKCS7Padding", "AES/CBC/PKCS7Padding",
"AES/CTR/NoPadding", "AES/CTR/NoPadding",
"AES/GCM/NoPadding"], "AES/GCM/NoPadding"],
(192): ["PBEWITHMD5AND192BITAES-CBC-OPENSSL", (192): ["PBEWITHMD5AND192BITAES-CBC-OPENSSL",
"PBEWITHSHA256AND192BITAES-CBC-BC", "PBEWITHSHA256AND192BITAES-CBC-BC",
"PBEWITHSHAAND192BITAES-CBC-BC", "PBEWITHSHAAND192BITAES-CBC-BC",
"AES/CBC/NoPadding",
"AES/CBC/PKCS7Padding", "AES/CBC/PKCS7Padding",
"AES/CTR/NoPadding", "AES/CTR/NoPadding",
"AES/GCM/NoPadding"], "AES/GCM/NoPadding"],
(256): ["PBEWITHMD5AND256BITAES-CBC-OPENSSL", (256): ["PBEWITHMD5AND256BITAES-CBC-OPENSSL",
"PBEWITHSHA256AND256BITAES-CBC-BC", "PBEWITHSHA256AND256BITAES-CBC-BC",
"PBEWITHSHAAND256BITAES-CBC-BC", "PBEWITHSHAAND256BITAES-CBC-BC",
"AES/CBC/NoPadding",
"AES/CBC/PKCS7Padding", "AES/CBC/PKCS7Padding",
"AES/CTR/NoPadding", "AES/CTR/NoPadding",
"AES/GCM/NoPadding"] "AES/GCM/NoPadding"]

View File

@ -35,6 +35,8 @@ import static org.junit.Assert.assertTrue
class PBKDF2CipherProviderGroovyTest { class PBKDF2CipherProviderGroovyTest {
private static final Logger logger = LoggerFactory.getLogger(PBKDF2CipherProviderGroovyTest.class) private static final Logger logger = LoggerFactory.getLogger(PBKDF2CipherProviderGroovyTest.class)
private static final String PLAINTEXT = "ExactBlockSizeRequiredForProcess"
private static List<EncryptionMethod> strongKDFEncryptionMethods private static List<EncryptionMethod> strongKDFEncryptionMethods
public static final String MICROBENCHMARK = "microbenchmark" public static final String MICROBENCHMARK = "microbenchmark"
@ -79,8 +81,6 @@ class PBKDF2CipherProviderGroovyTest {
final String PASSWORD = "shortPassword" final String PASSWORD = "shortPassword"
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -90,7 +90,7 @@ class PBKDF2CipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false)
@ -99,7 +99,7 @@ class PBKDF2CipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -140,8 +140,6 @@ class PBKDF2CipherProviderGroovyTest {
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final byte[] IV = Hex.decodeHex(IV_HEX as char[]) final byte[] IV = Hex.decodeHex(IV_HEX as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -150,7 +148,7 @@ class PBKDF2CipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false)
@ -159,7 +157,7 @@ class PBKDF2CipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -176,8 +174,6 @@ class PBKDF2CipherProviderGroovyTest {
final int LONG_KEY_LENGTH = 256 final int LONG_KEY_LENGTH = 256
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -187,7 +183,7 @@ class PBKDF2CipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false)
@ -196,7 +192,7 @@ class PBKDF2CipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -209,7 +205,6 @@ class PBKDF2CipherProviderGroovyTest {
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final byte[] IV = Hex.decodeHex(IV_HEX as char[]) final byte[] IV = Hex.decodeHex(IV_HEX as char[])
final String plaintext = "This is a plaintext message."
final EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC final EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
String prf = "" String prf = ""
@ -232,7 +227,6 @@ class PBKDF2CipherProviderGroovyTest {
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final byte[] IV = Hex.decodeHex(IV_HEX as char[]) final byte[] IV = Hex.decodeHex(IV_HEX as char[])
final String plaintext = "This is a plaintext message."
final EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC final EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
final PBKDF2CipherProvider SHA512_PROVIDER = new PBKDF2CipherProvider(DEFAULT_PRF, TEST_ITERATION_COUNT) final PBKDF2CipherProvider SHA512_PROVIDER = new PBKDF2CipherProvider(DEFAULT_PRF, TEST_ITERATION_COUNT)
@ -249,7 +243,7 @@ class PBKDF2CipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = SHA512_PROVIDER.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false) cipher = SHA512_PROVIDER.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false)
@ -258,7 +252,7 @@ class PBKDF2CipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
@Test @Test
@ -271,7 +265,6 @@ class PBKDF2CipherProviderGroovyTest {
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final byte[] IV = Hex.decodeHex(IV_HEX as char[]) final byte[] IV = Hex.decodeHex(IV_HEX as char[])
final String plaintext = "This is a plaintext message."
final EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC final EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
// Act // Act
@ -286,7 +279,7 @@ class PBKDF2CipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(encryptionMethod, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false)
@ -295,7 +288,7 @@ class PBKDF2CipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -334,7 +327,6 @@ class PBKDF2CipherProviderGroovyTest {
RandomIVPBECipherProvider sha256CP = new PBKDF2CipherProvider("SHA-256", TEST_ITERATION_COUNT) RandomIVPBECipherProvider sha256CP = new PBKDF2CipherProvider("SHA-256", TEST_ITERATION_COUNT)
RandomIVPBECipherProvider sha512CP = new PBKDF2CipherProvider("SHA-512", TEST_ITERATION_COUNT) RandomIVPBECipherProvider sha512CP = new PBKDF2CipherProvider("SHA-512", TEST_ITERATION_COUNT)
final String PLAINTEXT = "This is a plaintext message."
final String PASSWORD = "thisIsABadPassword" final String PASSWORD = "thisIsABadPassword"
final byte[] SALT = [0x11] * 16 final byte[] SALT = [0x11] * 16
final byte[] IV = [0x22] * 16 final byte[] IV = [0x22] * 16
@ -370,8 +362,6 @@ class PBKDF2CipherProviderGroovyTest {
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final byte[] IV = Hex.decodeHex(IV_HEX as char[]) final byte[] IV = Hex.decodeHex(IV_HEX as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -380,7 +370,7 @@ class PBKDF2CipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
def msg = shouldFail(IllegalArgumentException) { def msg = shouldFail(IllegalArgumentException) {
@ -426,8 +416,6 @@ class PBKDF2CipherProviderGroovyTest {
final byte[] SALT = Hex.decodeHex(SALT_HEX as char[]) final byte[] SALT = Hex.decodeHex(SALT_HEX as char[])
final byte[] IV = Hex.decodeHex(IV_HEX as char[]) final byte[] IV = Hex.decodeHex(IV_HEX as char[])
final String PLAINTEXT = "This is a plaintext message."
// Currently only AES ciphers are compatible with PBKDF2, so redundant to test all algorithms // Currently only AES ciphers are compatible with PBKDF2, so redundant to test all algorithms
final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC

View File

@ -46,6 +46,8 @@ import static org.junit.Assert.assertTrue
class ScryptCipherProviderGroovyTest { class ScryptCipherProviderGroovyTest {
private static final Logger logger = LoggerFactory.getLogger(ScryptCipherProviderGroovyTest.class) private static final Logger logger = LoggerFactory.getLogger(ScryptCipherProviderGroovyTest.class)
private static final String PLAINTEXT = "ExactBlockSizeRequiredForProcess"
private static List<EncryptionMethod> strongKDFEncryptionMethods private static List<EncryptionMethod> strongKDFEncryptionMethods
private static final int DEFAULT_KEY_LENGTH = 128 private static final int DEFAULT_KEY_LENGTH = 128
@ -88,8 +90,6 @@ class ScryptCipherProviderGroovyTest {
final String PASSWORD = "shortPassword" final String PASSWORD = "shortPassword"
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -99,7 +99,7 @@ class ScryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, DEFAULT_KEY_LENGTH, false)
@ -108,7 +108,7 @@ class ScryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -119,8 +119,6 @@ class ScryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("01" * 16 as char[]) final byte[] IV = Hex.decodeHex("01" * 16 as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -129,7 +127,7 @@ class ScryptCipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, false)
@ -138,7 +136,7 @@ class ScryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -153,8 +151,6 @@ class ScryptCipherProviderGroovyTest {
final int LONG_KEY_LENGTH = 256 final int LONG_KEY_LENGTH = 256
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -164,7 +160,7 @@ class ScryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false) cipher = cipherProvider.getCipher(em, PASSWORD, SALT, iv, LONG_KEY_LENGTH, false)
@ -173,7 +169,7 @@ class ScryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
} }
@ -265,7 +261,6 @@ class ScryptCipherProviderGroovyTest {
final String EXPECTED_FORMATTED_SALT = cipherProvider.formatSaltForScrypt(SALT) final String EXPECTED_FORMATTED_SALT = cipherProvider.formatSaltForScrypt(SALT)
logger.info("Expected salt: ${EXPECTED_FORMATTED_SALT}") logger.info("Expected salt: ${EXPECTED_FORMATTED_SALT}")
final String plaintext = "This is a plaintext message."
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
logger.info("Using algorithm: ${encryptionMethod.getAlgorithm()}") logger.info("Using algorithm: ${encryptionMethod.getAlgorithm()}")
@ -276,7 +271,7 @@ class ScryptCipherProviderGroovyTest {
byte[] iv = cipher.getIV() byte[] iv = cipher.getIV()
logger.info("IV: ${Hex.encodeHexString(iv)}") logger.info("IV: ${Hex.encodeHexString(iv)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
// Manually initialize a cipher for decrypt with the expected salt // Manually initialize a cipher for decrypt with the expected salt
@ -294,7 +289,7 @@ class ScryptCipherProviderGroovyTest {
logger.info("Recovered: ${recovered}") logger.info("Recovered: ${recovered}")
// Assert // Assert
assert plaintext.equals(recovered) assert PLAINTEXT.equals(recovered)
} }
@Test @Test
@ -368,8 +363,6 @@ class ScryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("00" * 16 as char[]) final byte[] IV = Hex.decodeHex("00" * 16 as char[])
final String plaintext = "This is a plaintext message."
// Act // Act
for (EncryptionMethod em : strongKDFEncryptionMethods) { for (EncryptionMethod em : strongKDFEncryptionMethods) {
logger.info("Using algorithm: ${em.getAlgorithm()}") logger.info("Using algorithm: ${em.getAlgorithm()}")
@ -378,7 +371,7 @@ class ScryptCipherProviderGroovyTest {
Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true) Cipher cipher = cipherProvider.getCipher(em, PASSWORD, SALT, IV, DEFAULT_KEY_LENGTH, true)
logger.info("IV: ${Hex.encodeHexString(IV)}") logger.info("IV: ${Hex.encodeHexString(IV)}")
byte[] cipherBytes = cipher.doFinal(plaintext.getBytes("UTF-8")) byte[] cipherBytes = cipher.doFinal(PLAINTEXT.getBytes("UTF-8"))
logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}") logger.info("Cipher text: ${Hex.encodeHexString(cipherBytes)} ${cipherBytes.length}")
def msg = shouldFail(IllegalArgumentException) { def msg = shouldFail(IllegalArgumentException) {
@ -398,8 +391,6 @@ class ScryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("01" * 16 as char[]) final byte[] IV = Hex.decodeHex("01" * 16 as char[])
final String PLAINTEXT = "This is a plaintext message."
final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS final def VALID_KEY_LENGTHS = AES_KEY_LENGTHS
EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC EncryptionMethod encryptionMethod = EncryptionMethod.AES_CBC
@ -431,8 +422,6 @@ class ScryptCipherProviderGroovyTest {
final byte[] SALT = cipherProvider.generateSalt() final byte[] SALT = cipherProvider.generateSalt()
final byte[] IV = Hex.decodeHex("00" * 16 as char[]) final byte[] IV = Hex.decodeHex("00" * 16 as char[])
final String PLAINTEXT = "This is a plaintext message."
// Even though Scrypt can derive keys of arbitrary length, it will fail to validate if the underlying cipher does not support it // Even though Scrypt can derive keys of arbitrary length, it will fail to validate if the underlying cipher does not support it
final def INVALID_KEY_LENGTHS = [-1, 40, 64, 112, 512] final def INVALID_KEY_LENGTHS = [-1, 40, 64, 112, 512]
// Currently only AES ciphers are compatible with Scrypt, so redundant to test all algorithms // Currently only AES ciphers are compatible with Scrypt, so redundant to test all algorithms

View File

@ -322,8 +322,8 @@ public class EncryptContent extends AbstractProcessor {
final String password = context.getProperty(PASSWORD).getValue(); final String password = context.getProperty(PASSWORD).getValue();
final KeyDerivationFunction kdf = KeyDerivationFunction.valueOf(context.getProperty(KEY_DERIVATION_FUNCTION).getValue()); final KeyDerivationFunction kdf = KeyDerivationFunction.valueOf(context.getProperty(KEY_DERIVATION_FUNCTION).getValue());
final String keyHex = context.getProperty(RAW_KEY_HEX).getValue(); final String keyHex = context.getProperty(RAW_KEY_HEX).getValue();
final boolean encrypt = context.getProperty(MODE).getValue().equalsIgnoreCase(ENCRYPT_MODE);
if (isPGPAlgorithm(algorithm)) { if (isPGPAlgorithm(algorithm)) {
final boolean encrypt = context.getProperty(MODE).getValue().equalsIgnoreCase(ENCRYPT_MODE);
final String publicKeyring = context.getProperty(PUBLIC_KEYRING).getValue(); final String publicKeyring = context.getProperty(PUBLIC_KEYRING).getValue();
final String publicUserId = context.getProperty(PUBLIC_KEY_USERID).getValue(); final String publicUserId = context.getProperty(PUBLIC_KEY_USERID).getValue();
final String privateKeyring = context.getProperty(PRIVATE_KEYRING).getValue(); final String privateKeyring = context.getProperty(PRIVATE_KEYRING).getValue();
@ -334,7 +334,7 @@ public class EncryptContent extends AbstractProcessor {
} else { // Not PGP } else { // Not PGP
boolean allowWeakCrypto = context.getProperty(ALLOW_WEAK_CRYPTO).getValue().equalsIgnoreCase(WEAK_CRYPTO_ALLOWED_NAME); boolean allowWeakCrypto = context.getProperty(ALLOW_WEAK_CRYPTO).getValue().equalsIgnoreCase(WEAK_CRYPTO_ALLOWED_NAME);
if (encryptionMethod.isKeyedCipher()) { // Raw key or derived key from password if (encryptionMethod.isKeyedCipher()) { // Raw key or derived key from password
validationResults.addAll(validateKeyed(encryptionMethod, kdf, keyHex, password, allowWeakCrypto)); validationResults.addAll(validateKeyed(encryptionMethod, kdf, keyHex, password, allowWeakCrypto, encrypt));
} else { // PBE } else { // PBE
validationResults.addAll(validatePBE(encryptionMethod, kdf, password, allowWeakCrypto)); validationResults.addAll(validatePBE(encryptionMethod, kdf, password, allowWeakCrypto));
} }
@ -479,7 +479,7 @@ public class EncryptContent extends AbstractProcessor {
} }
private List<ValidationResult> validateKeyed(EncryptionMethod encryptionMethod, KeyDerivationFunction kdf, String keyHex, String password, boolean allowWeakCrypto) { private List<ValidationResult> validateKeyed(EncryptionMethod encryptionMethod, KeyDerivationFunction kdf, String keyHex, String password, boolean allowWeakCrypto, boolean encrypt) {
List<ValidationResult> validationResults = new ArrayList<>(); List<ValidationResult> validationResults = new ArrayList<>();
boolean limitedStrengthCrypto = !CipherUtility.isUnlimitedStrengthCryptoSupported(); boolean limitedStrengthCrypto = !CipherUtility.isUnlimitedStrengthCryptoSupported();
@ -526,6 +526,17 @@ public class EncryptContent extends AbstractProcessor {
.explanation(KEY_DERIVATION_FUNCTION.getDisplayName() + " is required to be " + StringUtils.join(kdfsForKeyedCipher, ", ") + " when using algorithm " + .explanation(KEY_DERIVATION_FUNCTION.getDisplayName() + " is required to be " + StringUtils.join(kdfsForKeyedCipher, ", ") + " when using algorithm " +
encryptionMethod.getAlgorithm()).build()); encryptionMethod.getAlgorithm()).build());
} }
if (encrypt && EncryptionMethod.AES_CBC_NO_PADDING == encryptionMethod) {
validationResults.add(new ValidationResult.Builder()
.subject(ENCRYPTION_ALGORITHM.getDisplayName())
.input(encryptionMethod.name())
.explanation(String.format("Encryption not supported for [%s]", encryptionMethod.getAlgorithm()))
.valid(false)
.build()
);
}
return validationResults; return validationResults;
} }

View File

@ -25,6 +25,7 @@ import org.apache.nifi.security.util.KeyDerivationFunction
import org.apache.nifi.security.util.crypto.Argon2CipherProvider import org.apache.nifi.security.util.crypto.Argon2CipherProvider
import org.apache.nifi.security.util.crypto.Argon2SecureHasher import org.apache.nifi.security.util.crypto.Argon2SecureHasher
import org.apache.nifi.security.util.crypto.CipherUtility import org.apache.nifi.security.util.crypto.CipherUtility
import org.apache.nifi.security.util.crypto.KeyedEncryptor
import org.apache.nifi.security.util.crypto.PasswordBasedEncryptor import org.apache.nifi.security.util.crypto.PasswordBasedEncryptor
import org.apache.nifi.security.util.crypto.RandomIVPBECipherProvider import org.apache.nifi.security.util.crypto.RandomIVPBECipherProvider
import org.apache.nifi.util.MockFlowFile import org.apache.nifi.util.MockFlowFile
@ -58,6 +59,8 @@ class TestEncryptContentGroovy {
private static final String WEAK_CRYPTO_ALLOWED = EncryptContent.WEAK_CRYPTO_ALLOWED_NAME private static final String WEAK_CRYPTO_ALLOWED = EncryptContent.WEAK_CRYPTO_ALLOWED_NAME
private static final String WEAK_CRYPTO_NOT_ALLOWED = EncryptContent.WEAK_CRYPTO_NOT_ALLOWED_NAME private static final String WEAK_CRYPTO_NOT_ALLOWED = EncryptContent.WEAK_CRYPTO_NOT_ALLOWED_NAME
private static final List<EncryptionMethod> SUPPORTED_KEYED_ENCRYPTION_METHODS = EncryptionMethod.values().findAll { it.isKeyedCipher() && it != EncryptionMethod.AES_CBC_NO_PADDING }
@BeforeClass @BeforeClass
static void setUpOnce() throws Exception { static void setUpOnce() throws Exception {
Security.addProvider(new BouncyCastleProvider()) Security.addProvider(new BouncyCastleProvider())
@ -198,15 +201,13 @@ class TestEncryptContentGroovy {
Collection<ValidationResult> results Collection<ValidationResult> results
MockProcessContext pc MockProcessContext pc
def encryptionMethods = EncryptionMethod.values().findAll { it.isKeyedCipher() }
final int VALID_KEY_LENGTH = 128 final int VALID_KEY_LENGTH = 128
final String VALID_KEY_HEX = "ab" * (VALID_KEY_LENGTH / 8) final String VALID_KEY_HEX = "ab" * (VALID_KEY_LENGTH / 8)
logger.info("Using key ${VALID_KEY_HEX} (${VALID_KEY_HEX.length() * 4} bits)") logger.info("Using key ${VALID_KEY_HEX} (${VALID_KEY_HEX.length() * 4} bits)")
runner.setProperty(EncryptContent.MODE, EncryptContent.ENCRYPT_MODE) runner.setProperty(EncryptContent.MODE, EncryptContent.ENCRYPT_MODE)
encryptionMethods.each { EncryptionMethod encryptionMethod -> SUPPORTED_KEYED_ENCRYPTION_METHODS.each { EncryptionMethod encryptionMethod ->
logger.info("Trying encryption method ${encryptionMethod.name()}") logger.info("Trying encryption method ${encryptionMethod.name()}")
runner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, encryptionMethod.name()) runner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, encryptionMethod.name())
@ -315,8 +316,7 @@ class TestEncryptContentGroovy {
Collection<ValidationResult> results Collection<ValidationResult> results
MockProcessContext pc MockProcessContext pc
def keyedEncryptionMethods = EncryptionMethod.values().findAll { it.isKeyedCipher() } logger.info("Testing keyed encryption methods: ${SUPPORTED_KEYED_ENCRYPTION_METHODS*.name()}")
logger.info("Testing keyed encryption methods: ${keyedEncryptionMethods*.name()}")
final int VALID_KEY_LENGTH = 128 final int VALID_KEY_LENGTH = 128
final String VALID_KEY_HEX = "ab" * (VALID_KEY_LENGTH / 8) final String VALID_KEY_HEX = "ab" * (VALID_KEY_LENGTH / 8)
@ -330,7 +330,7 @@ class TestEncryptContentGroovy {
final def VALID_KDFS = KeyDerivationFunction.values().findAll { it.isStrongKDF() } final def VALID_KDFS = KeyDerivationFunction.values().findAll { it.isStrongKDF() }
// Scenario 1 - RKH w/ KDF NONE & em in [CBC, CTR, GCM] (no password) // Scenario 1 - RKH w/ KDF NONE & em in [CBC, CTR, GCM] (no password)
keyedEncryptionMethods.each { EncryptionMethod kem -> SUPPORTED_KEYED_ENCRYPTION_METHODS.each { EncryptionMethod kem ->
logger.info("Trying encryption method ${kem.name()} with KDF ${none.name()}") logger.info("Trying encryption method ${kem.name()} with KDF ${none.name()}")
runner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, kem.name()) runner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, kem.name())
runner.setProperty(EncryptContent.KEY_DERIVATION_FUNCTION, none.name()) runner.setProperty(EncryptContent.KEY_DERIVATION_FUNCTION, none.name())
@ -440,9 +440,7 @@ class TestEncryptContentGroovy {
testRunner.setProperty(EncryptContent.RAW_KEY_HEX, RAW_KEY_HEX) testRunner.setProperty(EncryptContent.RAW_KEY_HEX, RAW_KEY_HEX)
testRunner.setProperty(EncryptContent.KEY_DERIVATION_FUNCTION, KeyDerivationFunction.NONE.name()) testRunner.setProperty(EncryptContent.KEY_DERIVATION_FUNCTION, KeyDerivationFunction.NONE.name())
def keyedCipherEMs = EncryptionMethod.values().findAll { it.isKeyedCipher() } SUPPORTED_KEYED_ENCRYPTION_METHODS.each { EncryptionMethod encryptionMethod ->
keyedCipherEMs.each { EncryptionMethod encryptionMethod ->
logger.info("Attempting {}", encryptionMethod.name()) logger.info("Attempting {}", encryptionMethod.name())
testRunner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, encryptionMethod.name()) testRunner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, encryptionMethod.name())
testRunner.setProperty(EncryptContent.MODE, EncryptContent.ENCRYPT_MODE) testRunner.setProperty(EncryptContent.MODE, EncryptContent.ENCRYPT_MODE)
@ -469,6 +467,33 @@ class TestEncryptContentGroovy {
} }
} }
@Test
void testDecryptAesCbcNoPadding() {
final TestRunner testRunner = TestRunners.newTestRunner(new EncryptContent())
final String RAW_KEY_HEX = "ab" * 16
testRunner.setProperty(EncryptContent.RAW_KEY_HEX, RAW_KEY_HEX)
testRunner.setProperty(EncryptContent.KEY_DERIVATION_FUNCTION, KeyDerivationFunction.NONE.name())
testRunner.setProperty(EncryptContent.ENCRYPTION_ALGORITHM, EncryptionMethod.AES_CBC_NO_PADDING.name())
testRunner.setProperty(EncryptContent.MODE, EncryptContent.DECRYPT_MODE)
final String content = "ExactBlockSizeRequiredForProcess"
final byte[] bytes = content.getBytes(StandardCharsets.UTF_8)
final ByteArrayInputStream inputStream = new ByteArrayInputStream(bytes)
final ByteArrayOutputStream outputStream = new ByteArrayOutputStream()
final KeyedEncryptor encryptor = new KeyedEncryptor(EncryptionMethod.AES_CBC_NO_PADDING, Hex.decodeHex(RAW_KEY_HEX))
encryptor.encryptionCallback.process(inputStream, outputStream)
outputStream.close()
final byte[] encrypted = outputStream.toByteArray()
testRunner.enqueue(encrypted)
testRunner.run()
testRunner.assertAllFlowFilesTransferred(EncryptContent.REL_SUCCESS, 1)
MockFlowFile flowFile = testRunner.getFlowFilesForRelationship(EncryptContent.REL_SUCCESS).get(0)
flowFile.assertContentEquals(content)
}
// TODO: Implement // TODO: Implement
@Test @Test
void testArgon2EncryptionShouldWriteAttributesWithEncryptionMetadata() throws IOException { void testArgon2EncryptionShouldWriteAttributesWithEncryptionMetadata() throws IOException {