2021-05-21 15:43:06 +02:00
|
|
|
<?php
|
|
|
|
/* Icinga Web 2 | (c) 2021 Icinga GmbH | GPLv2+ */
|
|
|
|
|
|
|
|
namespace Icinga\Crypt;
|
|
|
|
|
|
|
|
use UnexpectedValueException;
|
|
|
|
use RuntimeException;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Data encryption and decryption using symmetric algorithm
|
|
|
|
*
|
|
|
|
* # Example Usage
|
|
|
|
*
|
|
|
|
* ```php
|
|
|
|
*
|
|
|
|
* // Encryption
|
2021-08-09 16:39:46 +02:00
|
|
|
* $encryptedData = (new AesCrypt())->encrypt($data); // Accepts a string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* // Encrypt and encode to Base64
|
|
|
|
* $encryptedData = (new AesCrypt())->encryptToBase64($data); // Accepts a string
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* // Decryption
|
|
|
|
* $aesCrypt = (new AesCrypt())
|
2021-08-09 16:39:46 +02:00
|
|
|
* ->setTag($tag) // if exists
|
2021-05-21 15:43:06 +02:00
|
|
|
* ->setIV($iv)
|
|
|
|
* ->setKey($key);
|
|
|
|
*
|
|
|
|
* $decryptedData = $aesCrypt->decrypt($data);
|
|
|
|
*
|
|
|
|
* // Decode from Base64 and decrypt
|
|
|
|
* $aesCrypt = (new AesCrypt())
|
|
|
|
* ->setTag($tag)
|
|
|
|
* ->setIV($iv)
|
|
|
|
* ->setKey($key);
|
|
|
|
*
|
2021-08-09 16:39:46 +02:00
|
|
|
* $decryptedData = $aesCrypt->decryptFromBase64($data);
|
2021-05-21 15:43:06 +02:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
class AesCrypt
|
|
|
|
{
|
2021-08-09 16:39:46 +02:00
|
|
|
/** @var array The list of cipher methods */
|
|
|
|
const METHODS = [
|
|
|
|
'aes-256-gcm',
|
|
|
|
'aes-256-cbc',
|
|
|
|
'aes-256-ctr'
|
|
|
|
];
|
|
|
|
|
2021-05-21 15:43:06 +02:00
|
|
|
/** @var string The encryption key */
|
|
|
|
private $key;
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
/** @var int The length of the key */
|
|
|
|
private $keyLength;
|
|
|
|
|
2021-05-21 15:43:06 +02:00
|
|
|
/** @var string The initialization vector which is not NULL */
|
|
|
|
private $iv;
|
|
|
|
|
|
|
|
/** @var string The authentication tag which is passed by reference when using AEAD cipher mode */
|
|
|
|
private $tag;
|
|
|
|
|
|
|
|
/** @var string The cipher method */
|
2021-08-09 16:39:46 +02:00
|
|
|
private $method;
|
2021-05-21 15:43:06 +02:00
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
public function __construct($keyLength = 128)
|
2021-05-21 15:43:06 +02:00
|
|
|
{
|
2021-08-09 16:39:46 +02:00
|
|
|
$this->keyLength = $keyLength;
|
2021-05-21 15:43:06 +02:00
|
|
|
}
|
|
|
|
|
2021-07-26 17:37:38 +02:00
|
|
|
/**
|
2021-08-09 16:39:46 +02:00
|
|
|
* Set the method
|
2021-07-26 17:37:38 +02:00
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setMethod($method)
|
|
|
|
{
|
|
|
|
$this->method = $method;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
/**
|
|
|
|
* Get the method
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getMethod()
|
|
|
|
{
|
|
|
|
if ($this->method === null) {
|
|
|
|
$this->method = $this->getSupportedMethod();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->method;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get supported method
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*
|
|
|
|
* @throws RuntimeException If none of the methods listed in the METHODS array is available
|
|
|
|
*/
|
|
|
|
protected function getSupportedMethod()
|
|
|
|
{
|
|
|
|
$availableMethods = openssl_get_cipher_methods();
|
|
|
|
$methods = self::METHODS;
|
|
|
|
|
|
|
|
if (! $this->isAuthenticatedEncryptionSupported()) {
|
|
|
|
unset($methods[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($methods as $method) {
|
|
|
|
if (in_array($method, $availableMethods)) {
|
|
|
|
return $method;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new RuntimeException('No supported method found');
|
|
|
|
}
|
|
|
|
|
2021-05-21 15:43:06 +02:00
|
|
|
/**
|
|
|
|
* Set the key
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setKey($key)
|
|
|
|
{
|
|
|
|
$this->key = $key;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the key
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
public function getKey()
|
|
|
|
{
|
|
|
|
if (empty($this->key)) {
|
2021-08-09 16:39:46 +02:00
|
|
|
$this->key = random_bytes($this->keyLength);
|
2021-05-21 15:43:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $this->key;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the IV
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setIV($iv)
|
|
|
|
{
|
|
|
|
$this->iv = $iv;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the IV
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
public function getIV()
|
|
|
|
{
|
|
|
|
if (empty($this->iv)) {
|
2021-08-09 16:39:46 +02:00
|
|
|
$len = openssl_cipher_iv_length($this->getMethod());
|
2021-07-26 17:37:38 +02:00
|
|
|
$this->iv = random_bytes($len);
|
2021-05-21 15:43:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $this->iv;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the Tag
|
|
|
|
*
|
|
|
|
* @return $this
|
2021-08-09 16:39:46 +02:00
|
|
|
*
|
|
|
|
* @throws RuntimeException If a tag is available but authenticated encryption (AE) is not supported.
|
|
|
|
*
|
|
|
|
* @throws UnexpectedValueException If tag length is less then 16
|
2021-05-21 15:43:06 +02:00
|
|
|
*/
|
|
|
|
public function setTag($tag)
|
|
|
|
{
|
2021-08-09 16:39:46 +02:00
|
|
|
if (! $this->isAuthenticatedEncryptionSupported()) {
|
|
|
|
throw new RuntimeException(sprintf(
|
|
|
|
"The given decryption method is not supported in php version '%s'",
|
|
|
|
PHP_VERSION
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2021-05-21 15:43:06 +02:00
|
|
|
if (strlen($tag) !== 16) {
|
|
|
|
throw new UnexpectedValueException(sprintf(
|
|
|
|
'expects tag length to be 16, got instead %s',
|
|
|
|
strlen($tag)
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->tag = $tag;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the Tag
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*
|
|
|
|
* @throws RuntimeException If the Tag is not set
|
|
|
|
*/
|
|
|
|
public function getTag()
|
|
|
|
{
|
|
|
|
if (empty($this->tag)) {
|
|
|
|
throw new RuntimeException('No tag set');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-08-09 16:39:46 +02:00
|
|
|
* Decrypt the given string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @param string $data
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*
|
|
|
|
* @throws RuntimeException If decryption fails
|
|
|
|
*/
|
|
|
|
public function decrypt($data)
|
|
|
|
{
|
2021-08-09 16:39:46 +02:00
|
|
|
if (! $this->isAuthenticatedEncryptionRequired()) {
|
|
|
|
return $this->nonAEDecrypt($data);
|
2021-07-26 17:37:38 +02:00
|
|
|
}
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
$decrypt = openssl_decrypt($data, $this->getMethod(), $this->getKey(), 0, $this->getIV(), $this->getTag());
|
2021-07-26 17:37:38 +02:00
|
|
|
|
|
|
|
if ($decrypt === false) {
|
2021-05-21 15:43:06 +02:00
|
|
|
throw new RuntimeException('Decryption failed');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $decrypt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-08-09 16:39:46 +02:00
|
|
|
* Decode from Base64 and decrypt the given string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @param string $data
|
|
|
|
*
|
2021-08-09 16:39:46 +02:00
|
|
|
* @return string The base64 decoded and decrypted string
|
|
|
|
*
|
|
|
|
* @deprecated Use decrypt() instead as it also returns a base64 decoded string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @throws RuntimeException If decryption fails
|
|
|
|
*/
|
|
|
|
public function decryptFromBase64($data)
|
|
|
|
{
|
|
|
|
return $this->decrypt(base64_decode($data));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-08-09 16:39:46 +02:00
|
|
|
* Encrypt the given string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @param string $data
|
|
|
|
*
|
2021-08-09 16:39:46 +02:00
|
|
|
* @return string encrypted string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @throws RuntimeException If decryption fails
|
|
|
|
*/
|
|
|
|
public function encrypt($data)
|
|
|
|
{
|
2021-08-09 16:39:46 +02:00
|
|
|
if (! $this->isAuthenticatedEncryptionRequired()) {
|
|
|
|
return $this->nonAEEncrypt($data);
|
2021-07-26 17:37:38 +02:00
|
|
|
}
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
$encrypt = openssl_encrypt($data, $this->getMethod(), $this->getKey(), 0, $this->getIV(), $this->tag);
|
2021-05-21 15:43:06 +02:00
|
|
|
|
2021-07-26 17:37:38 +02:00
|
|
|
if ($encrypt === false) {
|
2021-05-21 15:43:06 +02:00
|
|
|
throw new RuntimeException('Encryption failed');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $encrypt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-08-09 16:39:46 +02:00
|
|
|
* Encrypt the given string and encode to Base64
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @param string $data
|
|
|
|
*
|
2021-08-09 16:39:46 +02:00
|
|
|
* @return string encrypted and base64 encoded string
|
|
|
|
*
|
|
|
|
* @deprecated Use encrypt() instead as it also returns a base64 encoded string
|
2021-05-21 15:43:06 +02:00
|
|
|
*
|
|
|
|
* @throws RuntimeException If encryption fails
|
|
|
|
*/
|
|
|
|
public function encryptToBase64($data)
|
|
|
|
{
|
|
|
|
return base64_encode($this->encrypt($data));
|
|
|
|
}
|
2021-07-26 17:37:38 +02:00
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
/**
|
|
|
|
* Decrypt the given string with non Authenticated encryption (AE) cipher method
|
|
|
|
*
|
|
|
|
* @param string $data
|
|
|
|
*
|
|
|
|
* @return string decrypted string
|
|
|
|
*
|
|
|
|
* @throws RuntimeException If decryption fails
|
|
|
|
*/
|
|
|
|
private function nonAEDecrypt($data)
|
2021-07-26 17:37:38 +02:00
|
|
|
{
|
|
|
|
$c = base64_decode($data);
|
|
|
|
$hmac = substr($c, 0, 32);
|
|
|
|
$data = substr($c, 32);
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
$decrypt = openssl_decrypt($data, $this->getMethod(), $this->getKey(), 0, $this->getIV());
|
|
|
|
$calcHmac = hash_hmac('sha256', $this->getIV() . $data, $this->getKey(), true);
|
2021-07-26 17:37:38 +02:00
|
|
|
|
|
|
|
if ($decrypt === false || ! hash_equals($hmac, $calcHmac)) {
|
|
|
|
throw new RuntimeException('Decryption failed');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $decrypt;
|
|
|
|
}
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
/**
|
|
|
|
* Encrypt the given string with non Authenticated encryption (AE) cipher method
|
|
|
|
*
|
|
|
|
* @param string $data
|
|
|
|
*
|
|
|
|
* @return string encrypted string
|
|
|
|
*
|
|
|
|
* @throws RuntimeException If encryption fails
|
|
|
|
*/
|
|
|
|
private function nonAEEncrypt($data)
|
2021-07-26 17:37:38 +02:00
|
|
|
{
|
2021-08-09 16:39:46 +02:00
|
|
|
$encrypt = openssl_encrypt($data, $this->getMethod(), $this->getKey(), 0, $this->getIV());
|
2021-07-26 17:37:38 +02:00
|
|
|
|
|
|
|
if ($encrypt === false) {
|
|
|
|
throw new RuntimeException('Encryption failed');
|
|
|
|
}
|
|
|
|
|
2021-08-09 16:39:46 +02:00
|
|
|
$hmac = hash_hmac('sha256', $this->getIV() . $encrypt, $this->getKey(), true);
|
2021-07-26 17:37:38 +02:00
|
|
|
|
|
|
|
return base64_encode($hmac . $encrypt);
|
|
|
|
}
|
2021-08-09 16:39:46 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the Authenticated encryption (a tag) is required
|
|
|
|
*
|
|
|
|
* @return bool True if required false otherwise
|
|
|
|
*/
|
|
|
|
public function isAuthenticatedEncryptionRequired()
|
|
|
|
{
|
|
|
|
return $this->getMethod() === 'aes-256-gcm';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the php version supports Authenticated encryption (AE) or not
|
|
|
|
*
|
|
|
|
* @return bool True if supported false otherwise
|
|
|
|
*/
|
|
|
|
public function isAuthenticatedEncryptionSupported()
|
|
|
|
{
|
|
|
|
return PHP_VERSION_ID >= 70100;
|
|
|
|
}
|
2021-05-21 15:43:06 +02:00
|
|
|
}
|