31) { trigger_error(__d( 'cake_dev', 'Invalid value, cost must be between %s and %s', array(4, 31) ), E_USER_WARNING); return null; } static::$hashCost = $cost; } /** * Get random bytes from a secure source. * * This method will fall back to an insecure source and trigger a warning, * if it cannot find a secure source of random data. * * @param int $length The number of bytes you want. * @return string Random bytes in binary. */ public static function randomBytes($length) { if (function_exists('random_bytes')) { return random_bytes($length); } if (function_exists('openssl_random_pseudo_bytes')) { return openssl_random_pseudo_bytes($length); } if (function_exists('mcrypt_create_iv')) { return mcrypt_create_iv($length, MCRYPT_DEV_URANDOM); } trigger_error( 'You do not have a safe source of random data available. ' . 'Install either the openssl extension, the mcrypt extension, or paragonie/random_compat. ' . 'Falling back to an insecure random source.', E_USER_WARNING ); $bytes = ''; $byteLength = 0; while ($byteLength < $length) { $bytes .= static::hash(CakeText::uuid() . uniqid(mt_rand(), true), 'sha512', true); $byteLength = strlen($bytes); } return substr($bytes, 0, $length); } /** * Runs $text through a XOR cipher. * * *Note* This is not a cryptographically strong method and should not be used * for sensitive data. Additionally this method does *not* work in environments * where suhosin is enabled. * * Instead you should use Security::encrypt() when you need strong * encryption. * * @param string $text Encrypted string to decrypt, normal string to encrypt * @param string $key Key to use * @return string Encrypted/Decrypted string * @deprecated 3.0.0 Will be removed in 3.0. */ public static function cipher($text, $key) { if (empty($key)) { trigger_error(__d('cake_dev', 'You cannot use an empty key for %s', 'Security::cipher()'), E_USER_WARNING); return ''; } srand((int)(float)Configure::read('Security.cipherSeed')); $out = ''; $keyLength = strlen($key); for ($i = 0, $textLength = strlen($text); $i < $textLength; $i++) { $j = ord(substr($key, $i % $keyLength, 1)); while ($j--) { rand(0, 255); } $mask = rand(0, 255); $out .= chr(ord(substr($text, $i, 1)) ^ $mask); } srand(); return $out; } /** * Encrypts/Decrypts a text using the given key using rijndael method. * * Prior to 2.3.1, a fixed initialization vector was used. This was not * secure. This method now uses a random iv, and will silently upgrade values when * they are re-encrypted. * * @param string $text Encrypted string to decrypt, normal string to encrypt * @param string $key Key to use as the encryption key for encrypted data. * @param string $operation Operation to perform, encrypt or decrypt * @return string Encrypted/Decrypted string */ public static function rijndael($text, $key, $operation) { if (empty($key)) { trigger_error(__d('cake_dev', 'You cannot use an empty key for %s', 'Security::rijndael()'), E_USER_WARNING); return ''; } if (empty($operation) || !in_array($operation, array('encrypt', 'decrypt'))) { trigger_error(__d('cake_dev', 'You must specify the operation for Security::rijndael(), either encrypt or decrypt'), E_USER_WARNING); return ''; } if (strlen($key) < 32) { trigger_error(__d('cake_dev', 'You must use a key larger than 32 bytes for Security::rijndael()'), E_USER_WARNING); return ''; } $algorithm = MCRYPT_RIJNDAEL_256; $mode = MCRYPT_MODE_CBC; $ivSize = mcrypt_get_iv_size($algorithm, $mode); $cryptKey = substr($key, 0, 32); if ($operation === 'encrypt') { $iv = mcrypt_create_iv($ivSize, MCRYPT_RAND); return $iv . '$$' . mcrypt_encrypt($algorithm, $cryptKey, $text, $mode, $iv); } // Backwards compatible decrypt with fixed iv if (substr($text, $ivSize, 2) !== '$$') { $iv = substr($key, strlen($key) - 32, 32); return rtrim(mcrypt_decrypt($algorithm, $cryptKey, $text, $mode, $iv), "\0"); } $iv = substr($text, 0, $ivSize); $text = substr($text, $ivSize + 2); return rtrim(mcrypt_decrypt($algorithm, $cryptKey, $text, $mode, $iv), "\0"); } /** * Generates a pseudo random salt suitable for use with php's crypt() function. * The salt length should not exceed 27. The salt will be composed of * [./0-9A-Za-z]{$length}. * * @param int $length The length of the returned salt * @return string The generated salt */ protected static function _salt($length = 22) { $salt = str_replace( array('+', '='), '.', base64_encode(sha1(uniqid(Configure::read('Security.salt'), true), true)) ); return substr($salt, 0, $length); } /** * One way encryption using php's crypt() function. To use blowfish hashing see ``Security::hash()`` * * @param string $password The string to be encrypted. * @param mixed $salt false to generate a new salt or an existing salt. * @return string The hashed string or an empty string on error. */ protected static function _crypt($password, $salt = false) { if ($salt === false || $salt === null || $salt === '') { $salt = static::_salt(22); $salt = vsprintf('$2a$%02d$%s', array(static::$hashCost, $salt)); } $invalidCipher = ( strpos($salt, '$2y$') !== 0 && strpos($salt, '$2x$') !== 0 && strpos($salt, '$2a$') !== 0 ); if ($salt === true || $invalidCipher || strlen($salt) < 29) { trigger_error(__d( 'cake_dev', 'Invalid salt: %s for %s Please visit http://www.php.net/crypt and read the appropriate section for building %s salts.', array($salt, 'blowfish', 'blowfish') ), E_USER_WARNING); return ''; } return crypt($password, $salt); } /** * Encrypt a value using AES-256. * * *Caveat* You cannot properly encrypt/decrypt data with trailing null bytes. * Any trailing null bytes will be removed on decryption due to how PHP pads messages * with nulls prior to encryption. * * @param string $plain The value to encrypt. * @param string $key The 256 bit/32 byte key to use as a cipher key. * @param string $hmacSalt The salt to use for the HMAC process. Leave null to use Security.salt. * @return string Encrypted data. * @throws CakeException On invalid data or key. */ public static function encrypt($plain, $key, $hmacSalt = null) { static::_checkKey($key, 'encrypt()'); if ($hmacSalt === null) { $hmacSalt = Configure::read('Security.salt'); } // Generate the encryption and hmac key. $key = substr(hash('sha256', $key . $hmacSalt), 0, 32); if (static::engine() === 'openssl') { $method = 'AES-256-CBC'; $ivSize = openssl_cipher_iv_length($method); $iv = openssl_random_pseudo_bytes($ivSize); $padLength = (int)ceil((strlen($plain) ?: 1) / $ivSize) * $ivSize; $ciphertext = openssl_encrypt(str_pad($plain, $padLength, "\0"), $method, $key, true, $iv); // Remove the PKCS#7 padding block for compatibility with mcrypt. // Since we have padded the provided data with \0, the final block contains only padded bytes. // So it can be removed safely. $ciphertext = $iv . substr($ciphertext, 0, -$ivSize); } else { $algorithm = MCRYPT_RIJNDAEL_128; $mode = MCRYPT_MODE_CBC; $ivSize = mcrypt_get_iv_size($algorithm, $mode); $iv = mcrypt_create_iv($ivSize, MCRYPT_DEV_URANDOM); $ciphertext = $iv . mcrypt_encrypt($algorithm, $key, $plain, $mode, $iv); } $hmac = hash_hmac('sha256', $ciphertext, $key); return $hmac . $ciphertext; } /** * Check the encryption key for proper length. * * @param string $key Key to check. * @param string $method The method the key is being checked for. * @return void * @throws CakeException When key length is not 256 bit/32 bytes */ protected static function _checkKey($key, $method) { if (strlen($key) < 32) { throw new CakeException(__d('cake_dev', 'Invalid key for %s, key must be at least 256 bits (32 bytes) long.', $method)); } } /** * Decrypt a value using AES-256. * * @param string $cipher The ciphertext to decrypt. * @param string $key The 256 bit/32 byte key to use as a cipher key. * @param string $hmacSalt The salt to use for the HMAC process. Leave null to use Security.salt. * @return string Decrypted data. Any trailing null bytes will be removed. * @throws CakeException On invalid data or key. */ public static function decrypt($cipher, $key, $hmacSalt = null) { static::_checkKey($key, 'decrypt()'); if (empty($cipher)) { throw new CakeException(__d('cake_dev', 'The data to decrypt cannot be empty.')); } if ($hmacSalt === null) { $hmacSalt = Configure::read('Security.salt'); } // Generate the encryption and hmac key. $key = substr(hash('sha256', $key . $hmacSalt), 0, 32); // Split out hmac for comparison $macSize = 64; $hmac = substr($cipher, 0, $macSize); $cipher = substr($cipher, $macSize); $compareHmac = hash_hmac('sha256', $cipher, $key); if ($hmac !== $compareHmac) { return false; } if (static::engine() === 'openssl') { $method = 'AES-256-CBC'; $ivSize = openssl_cipher_iv_length($method); $iv = substr($cipher, 0, $ivSize); $cipher = substr($cipher, $ivSize); // Regenerate PKCS#7 padding block $padding = openssl_encrypt('', $method, $key, true, substr($cipher, -$ivSize)); $plain = openssl_decrypt($cipher . $padding, $method, $key, true, $iv); } else { $algorithm = MCRYPT_RIJNDAEL_128; $mode = MCRYPT_MODE_CBC; $ivSize = mcrypt_get_iv_size($algorithm, $mode); $iv = substr($cipher, 0, $ivSize); $cipher = substr($cipher, $ivSize); $plain = mcrypt_decrypt($algorithm, $key, $cipher, $mode, $iv); } return rtrim($plain, "\0"); } /** * Set or get the encryption engine * * @param string $engine The encryption engine to use * @return string */ public static function engine($engine = null) { if (func_num_args() > 0) { static::$_engine = $engine; } elseif (static::$_engine === null) { static::$_engine = 'mcrypt'; if (!extension_loaded('mcrypt') && extension_loaded('openssl') && version_compare(PHP_VERSION, '5.3.3', '>=')) { static::$_engine = 'openssl'; } } return static::$_engine; } }