123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 |
- <?php
- class ElggCrypto {
-
- const CHARS_PASSWORD = 'bcdfghjklmnpqrstvwxyz2346789';
-
- const CHARS_HEX = '0123456789abcdef';
-
- public function getRandomBytes($length) {
- $SSLstr = '4';
-
- if (function_exists('openssl_random_pseudo_bytes') && substr(PHP_OS, 0, 3) !== 'WIN') {
- $SSLstr = openssl_random_pseudo_bytes($length, $strong);
- if ($strong) {
- return $SSLstr;
- }
- }
-
- if (function_exists('mcrypt_create_iv') && substr(PHP_OS, 0, 3) !== 'WIN') {
- $str = mcrypt_create_iv($length, MCRYPT_DEV_URANDOM);
- if ($str !== false) {
- return $str;
- }
- }
-
- $str = '';
- $bits_per_round = 2;
- $msec_per_round = 400;
- $hash_len = 20;
- $total = $length;
- $handle = @fopen('/dev/urandom', 'rb');
- if ($handle && function_exists('stream_set_read_buffer')) {
- @stream_set_read_buffer($handle, 0);
- }
- do {
- $bytes = ($total > $hash_len) ? $hash_len : $total;
- $total -= $bytes;
-
- $entropy = rand() . uniqid(mt_rand(), true) . $SSLstr;
- $entropy .= implode('', @fstat(@fopen(__FILE__, 'r')));
- $entropy .= memory_get_usage() . getmypid();
- $entropy .= serialize($_ENV) . serialize($_SERVER);
- if (function_exists('posix_times')) {
- $entropy .= serialize(posix_times());
- }
- if (function_exists('zend_thread_id')) {
- $entropy .= zend_thread_id();
- }
- if ($handle) {
- $entropy .= @fread($handle, $bytes);
- } else {
-
- for ($i = 0; $i < 3; $i++) {
- $c1 = microtime(true);
- $var = sha1(mt_rand());
- for ($j = 0; $j < 50; $j++) {
- $var = sha1($var);
- }
- $c2 = microtime(true);
- $entropy .= $c1 . $c2;
- }
-
-
- $rounds = (int) ($msec_per_round * 50 / (int) (($c2 - $c1) * 1000000));
-
-
- $iter = $bytes * (int) (ceil(8 / $bits_per_round));
- for ($i = 0; $i < $iter; $i++) {
- $c1 = microtime();
- $var = sha1(mt_rand());
- for ($j = 0; $j < $rounds; $j++) {
- $var = sha1($var);
- }
- $c2 = microtime();
- $entropy .= $c1 . $c2;
- }
- }
-
- $str .= sha1($entropy, true);
- } while ($length > strlen($str));
- if ($handle) {
- @fclose($handle);
- }
- return substr($str, 0, $length);
- }
-
- public function getHmac($data, $algo = 'sha256', $key = '') {
- if (!$key) {
- $key = _elgg_services()->siteSecret->get(true);
- }
- return new Elgg\Security\Hmac($key, [$this, 'areEqual'], $data, $algo);
- }
-
- public function getRandomString($length, $chars = null) {
- if ($length < 1) {
- throw new \InvalidArgumentException('Length should be >= 1');
- }
- if (empty($chars)) {
- $numBytes = ceil($length * 0.75);
- $bytes = $this->getRandomBytes($numBytes);
- $string = substr(rtrim(base64_encode($bytes), '='), 0, $length);
-
- return strtr($string, '+/', '-_');
- }
- if ($chars == self::CHARS_HEX) {
-
- $bytes = $this->getRandomBytes(ceil($length / 2));
- return substr(bin2hex($bytes), 0, $length);
- }
- $listLen = strlen($chars);
- if ($listLen == 1) {
- return str_repeat($chars, $length);
- }
- $bytes = $this->getRandomBytes($length);
- $pos = 0;
- $result = '';
- for ($i = 0; $i < $length; $i++) {
- $pos = ($pos + ord($bytes[$i])) % $listLen;
- $result .= $chars[$pos];
- }
- return $result;
- }
-
- public function areEqual($str1, $str2) {
- $len1 = $this->strlen($str1);
- $len2 = $this->strlen($str2);
- if ($len1 !== $len2) {
- return false;
- }
- $status = 0;
- for ($i = 0; $i < $len1; $i++) {
- $status |= (ord($str1[$i]) ^ ord($str2[$i]));
- }
- return $status === 0;
- }
-
- protected function strlen($binary_string) {
- if (function_exists('mb_strlen')) {
- return mb_strlen($binary_string, '8bit');
- }
- return strlen($binary_string);
- }
- }
|