Как использовать PHP password_hash для хеширования и проверки паролей

94

Недавно я пытался реализовать свою собственную безопасность в сценарии входа в систему, на который наткнулся в Интернете. Изо всех сил пытаясь научиться создавать собственный сценарий для генерации соли для каждого пользователя, я наткнулся на него password_hash.

Насколько я понимаю (основываясь на чтении на этой странице ), соль уже генерируется в строке, когда вы используете password_hash. Это правда?

Еще один вопрос, который у меня возник: не было бы разумно иметь 2 соли? Один прямо в файле, а другой в БД? Таким образом, если кто-то скомпрометирует вашу соль в БД, она останется прямо в файле? Я читал здесь, что хранение солей никогда не было разумной идеей, но меня всегда смущало, что люди имели в виду под этим.

Джош Поттер
источник
8
Нет. Пусть функция позаботится о соли. Двойной посол доставит вам неприятности, и в этом нет необходимости.
Funk Forty Niner

Ответы:

182

Использование password_hash- рекомендуемый способ хранения паролей. Не разделяйте их на БД и файлы.

Допустим, у нас есть следующий ввод:

$password = $_POST['password'];

Сначала вы хешируете пароль, делая это:

$hashed_password = password_hash($password, PASSWORD_DEFAULT);

Затем посмотрите результат:

var_dump($hashed_password);

Как видите, он хеширован. (Я предполагаю, что вы сделали эти шаги).

Теперь вы сохраняете этот хешированный пароль в своей базе данных, следя за тем , чтобы столбец пароля был достаточно большим для хранения хешированного значения (не менее 60 символов или больше) . Когда пользователь просит войти в систему, вы проверяете ввод пароля с этим значением хеш-функции в базе данных, выполнив следующие действия:

// Query the database for username and password
// ...

if(password_verify($password, $hashed_password)) {
    // If the password inputs matched the hashed password in the database
    // Do something, you know... log them in.
} 

// Else, Redirect them back to the login page.

Официальная ссылка

Акар
источник
2
Хорошо, я просто попробовал это, и это сработало. Я сомневался в этой функции, потому что она казалась слишком простой. Как долго вы рекомендуете делать длину своего варчара? 225?
Джош Поттер
4
Это уже есть в руководствах php.net/manual/en/function.password-hash.php --- php.net/manual/en/function.password-verify.php, которые OP, вероятно, не прочитал или не понял. Этот вопрос задают чаще, чем ничего.
Funk Forty Niner
Это другая страница.
Джош Поттер
@JoshPotter отличается от чего? плюс, заметили, что они не ответили на ваш второй вопрос. они, вероятно, ожидают, что вы это узнаете сами, или они не знают.
Funk Forty Niner
8
@FunkFortyNiner, b / c Джош задал вопрос, я нашел его, 2 года спустя, и мне это помогло. В этом суть ТАК. Это руководство ясно как грязь.
toddmo
23

Да, вы правильно поняли, функция password_hash () сама сгенерирует соль и включит ее в полученное хеш-значение. Хранить соль в базе данных абсолютно правильно, она выполняет свою работу, даже если известна.

// Hash a new password for storing in the database.
// The function automatically generates a cryptographically safe salt.
$hashToStoreInDb = password_hash($_POST['password'], PASSWORD_DEFAULT);

// Check if the hash of the entered login password, matches the stored hash.
// The salt and the cost factor will be extracted from $existingHashFromDb.
$isPasswordCorrect = password_verify($_POST['password'], $existingHashFromDb);

Вторая соль, которую вы упомянули (та, что хранится в файле), на самом деле является перцем или ключом на стороне сервера. Если вы добавляете его перед перемешиванием (как соль), то вы добавляете перец. Однако есть способ получше: сначала вы можете вычислить хэш, а затем зашифровать (двусторонним) хеш серверным ключом. Это дает вам возможность при необходимости менять ключ.

В отличие от соли этот ключ следует держать в секрете. Люди часто перемешивают и пытаются спрятать соль, но лучше дать соли сделать свое дело и добавить секрет ключом.

Мартинстекли
источник
8

Да, это правда. Почему вы сомневаетесь в php faq по функции? :)

Результат бега password_hash()состоит из четырех частей:

  1. используемый алгоритм
  2. параметры
  3. соль
  4. фактический хеш пароля

Как видите, хеш - это его часть.

Конечно, у вас может быть дополнительная соль для дополнительного уровня безопасности, но я, честно говоря, думаю, что это излишество в обычном приложении php. Алгоритм bcrypt по умолчанию хорош, а дополнительный алгоритм blowfish, возможно, даже лучше.

Джоэл Хинц
источник
2
BCrypt - это функция хеширования , а Blowfish - алгоритм шифрования . Однако BCrypt происходит от алгоритма Blowfish.
martinstoeckli
7

Никогда не используйте md5 () для защиты пароля, даже с солью, это всегда опасно !!

Защитите свой пароль с помощью новейших алгоритмов хеширования, как показано ниже.

<?php

// Your original Password
$password = '121@121';

//PASSWORD_BCRYPT or PASSWORD_DEFAULT use any in the 2nd parameter
/*
PASSWORD_BCRYPT always results 60 characters long string.
PASSWORD_DEFAULT capacity is beyond 60 characters
*/
$password_encrypted = password_hash($password, PASSWORD_BCRYPT);

Для сопоставления с зашифрованным паролем базы данных и паролем, введенным пользователем, используйте следующую функцию.

<?php 

if (password_verify($password_inputted_by_user, $password_encrypted)) {
    // Success!
    echo 'Password Matches';
}else {
    // Invalid credentials
    echo 'Password Mismatch';
}

Если вы хотите использовать свою собственную соль, используйте для этого собственную сгенерированную функцию, просто следуйте инструкциям ниже, но я не рекомендую это, поскольку в последних версиях PHP она считается устаревшей.

Прочтите о password_hash () перед использованием кода ниже.

<?php

$options = [
    'salt' => your_custom_function_for_salt(), 
    //write your own code to generate a suitable & secured salt
    'cost' => 12 // the default cost is 10
];

$hash = password_hash($your_password, PASSWORD_DEFAULT, $options);
Махеш Ядав
источник
4
Опция соли устарела по веским причинам, потому что функция делает все возможное, чтобы генерировать криптографически безопасную соль, и сделать это лучше практически невозможно.
martinstoeckli 01
@martinstoeckli, да, вы правы, я только что обновил свой ответ, спасибо!
Махеш Ядав
if (isset ($ _ POST ['btn-signup'])) {$ uname = mysql_real_escape_string ($ _ POST ['uname']); $ email = mysql_real_escape_string ($ _ POST ['электронная почта']); $ upass = md5 (mysql_real_escape_string ($ _ POST ['пройти'])); Это код, используемый в login.php .. я хочу обойтись без escape и md5. я хочу использовать хеш пароля ..
rashmi sm
PASSWORD_DEFAULT - использовать алгоритм bcrypt (требуется PHP 5.5.0). Обратите внимание, что эта константа должна изменяться с течением времени по мере добавления в PHP новых и более сильных алгоритмов. По этой причине длина результата использования этого идентификатора может меняться со временем.
Адриан П.
5

Существует явное отсутствие обсуждения обратной и прямой совместимости, которая встроена в функции паролей PHP. В частности:

  1. Обратная совместимость: функции паролей, по сути, представляют собой хорошо написанную оболочку crypt()и по своей сути обратно совместимы с crypt()хешами -format, даже если они используют устаревшие и / или небезопасные хеш-алгоритмы.
  2. Нападающие совместимость: Установка password_needs_rehash()и немного логики в вашей аутентификации рабочий процесс может держать вас в ваших хэшей в курсе текущих и будущих алгоритмов с потенциально нулевыми будущих изменений в рабочий процесс. Примечание. Любая строка, не соответствующая указанному алгоритму, будет помечена как требующая повторного хеширования, включая хеши, не поддерживающие шифрование.

Например:

class FakeDB {
    public function __call($name, $args) {
        printf("%s::%s(%s)\n", __CLASS__, $name, json_encode($args));
        return $this;
    }
}

class MyAuth {
    protected $dbh;
    protected $fakeUsers = [
        // old crypt-md5 format
        1 => ['password' => '$1$AVbfJOzY$oIHHCHlD76Aw1xmjfTpm5.'],
        // old salted md5 format
        2 => ['password' => '3858f62230ac3c915f300c664312c63f', 'salt' => 'bar'],
        // current bcrypt format
        3 => ['password' => '$2y$10$3eUn9Rnf04DR.aj8R3WbHuBO9EdoceH9uKf6vMiD7tz766rMNOyTO']
    ];

    public function __construct($dbh) {
        $this->dbh = $dbh;
    }

    protected function getuser($id) {
        // just pretend these are coming from the DB
        return $this->fakeUsers[$id];
    }

    public function authUser($id, $password) {
        $userInfo = $this->getUser($id);

        // Do you have old, turbo-legacy, non-crypt hashes?
        if( strpos( $userInfo['password'], '$' ) !== 0 ) {
            printf("%s::legacy_hash\n", __METHOD__);
            $res = $userInfo['password'] === md5($password . $userInfo['salt']);
        } else {
            printf("%s::password_verify\n", __METHOD__);
            $res = password_verify($password, $userInfo['password']);
        }

        // once we've passed validation we can check if the hash needs updating.
        if( $res && password_needs_rehash($userInfo['password'], PASSWORD_DEFAULT) ) {
            printf("%s::rehash\n", __METHOD__);
            $stmt = $this->dbh->prepare('UPDATE users SET pass = ? WHERE user_id = ?');
            $stmt->execute([password_hash($password, PASSWORD_DEFAULT), $id]);
        }

        return $res;
    }
}

$auth = new MyAuth(new FakeDB());

for( $i=1; $i<=3; $i++) {
    var_dump($auth->authuser($i, 'foo'));
    echo PHP_EOL;
}

Выход:

MyAuth::authUser::password_verify
MyAuth::authUser::rehash
FakeDB::prepare(["UPDATE users SET pass = ? WHERE user_id = ?"])
FakeDB::execute([["$2y$10$zNjPwqQX\/RxjHiwkeUEzwOpkucNw49yN4jjiRY70viZpAx5x69kv.",1]])
bool(true)

MyAuth::authUser::legacy_hash
MyAuth::authUser::rehash
FakeDB::prepare(["UPDATE users SET pass = ? WHERE user_id = ?"])
FakeDB::execute([["$2y$10$VRTu4pgIkGUvilTDRTXYeOQSEYqe2GjsPoWvDUeYdV2x\/\/StjZYHu",2]])
bool(true)

MyAuth::authUser::password_verify
bool(true)

В качестве последнего примечания, учитывая, что вы можете повторно хешировать пароль пользователя только при входе в систему, вам следует подумать о «отключении» небезопасных устаревших хэшей для защиты ваших пользователей. Под этим я подразумеваю, что по истечении определенного периода отсрочки вы удаляете все небезопасные (например, голые MD5 / SHA / в противном случае слабые) хэши и заставляете пользователей полагаться на механизмы сброса пароля вашего приложения.

Саммитч
источник
0

Полный код пароля класса:

Class Password {

    public function __construct() {}


    /**
     * Hash the password using the specified algorithm
     *
     * @param string $password The password to hash
     * @param int    $algo     The algorithm to use (Defined by PASSWORD_* constants)
     * @param array  $options  The options for the algorithm to use
     *
     * @return string|false The hashed password, or false on error.
     */
    function password_hash($password, $algo, array $options = array()) {
        if (!function_exists('crypt')) {
            trigger_error("Crypt must be loaded for password_hash to function", E_USER_WARNING);
            return null;
        }
        if (!is_string($password)) {
            trigger_error("password_hash(): Password must be a string", E_USER_WARNING);
            return null;
        }
        if (!is_int($algo)) {
            trigger_error("password_hash() expects parameter 2 to be long, " . gettype($algo) . " given", E_USER_WARNING);
            return null;
        }
        switch ($algo) {
            case PASSWORD_BCRYPT :
                // Note that this is a C constant, but not exposed to PHP, so we don't define it here.
                $cost = 10;
                if (isset($options['cost'])) {
                    $cost = $options['cost'];
                    if ($cost < 4 || $cost > 31) {
                        trigger_error(sprintf("password_hash(): Invalid bcrypt cost parameter specified: %d", $cost), E_USER_WARNING);
                        return null;
                    }
                }
                // The length of salt to generate
                $raw_salt_len = 16;
                // The length required in the final serialization
                $required_salt_len = 22;
                $hash_format = sprintf("$2y$%02d$", $cost);
                break;
            default :
                trigger_error(sprintf("password_hash(): Unknown password hashing algorithm: %s", $algo), E_USER_WARNING);
                return null;
        }
        if (isset($options['salt'])) {
            switch (gettype($options['salt'])) {
                case 'NULL' :
                case 'boolean' :
                case 'integer' :
                case 'double' :
                case 'string' :
                    $salt = (string)$options['salt'];
                    break;
                case 'object' :
                    if (method_exists($options['salt'], '__tostring')) {
                        $salt = (string)$options['salt'];
                        break;
                    }
                case 'array' :
                case 'resource' :
                default :
                    trigger_error('password_hash(): Non-string salt parameter supplied', E_USER_WARNING);
                    return null;
            }
            if (strlen($salt) < $required_salt_len) {
                trigger_error(sprintf("password_hash(): Provided salt is too short: %d expecting %d", strlen($salt), $required_salt_len), E_USER_WARNING);
                return null;
            } elseif (0 == preg_match('#^[a-zA-Z0-9./]+$#D', $salt)) {
                $salt = str_replace('+', '.', base64_encode($salt));
            }
        } else {
            $salt = str_replace('+', '.', base64_encode($this->generate_entropy($required_salt_len)));
        }
        $salt = substr($salt, 0, $required_salt_len);

        $hash = $hash_format . $salt;

        $ret = crypt($password, $hash);

        if (!is_string($ret) || strlen($ret) <= 13) {
            return false;
        }

        return $ret;
    }


    /**
     * Generates Entropy using the safest available method, falling back to less preferred methods depending on support
     *
     * @param int $bytes
     *
     * @return string Returns raw bytes
     */
    function generate_entropy($bytes){
        $buffer = '';
        $buffer_valid = false;
        if (function_exists('mcrypt_create_iv') && !defined('PHALANGER')) {
            $buffer = mcrypt_create_iv($bytes, MCRYPT_DEV_URANDOM);
            if ($buffer) {
                $buffer_valid = true;
            }
        }
        if (!$buffer_valid && function_exists('openssl_random_pseudo_bytes')) {
            $buffer = openssl_random_pseudo_bytes($bytes);
            if ($buffer) {
                $buffer_valid = true;
            }
        }
        if (!$buffer_valid && is_readable('/dev/urandom')) {
            $f = fopen('/dev/urandom', 'r');
            $read = strlen($buffer);
            while ($read < $bytes) {
                $buffer .= fread($f, $bytes - $read);
                $read = strlen($buffer);
            }
            fclose($f);
            if ($read >= $bytes) {
                $buffer_valid = true;
            }
        }
        if (!$buffer_valid || strlen($buffer) < $bytes) {
            $bl = strlen($buffer);
            for ($i = 0; $i < $bytes; $i++) {
                if ($i < $bl) {
                    $buffer[$i] = $buffer[$i] ^ chr(mt_rand(0, 255));
                } else {
                    $buffer .= chr(mt_rand(0, 255));
                }
            }
        }
        return $buffer;
    }

    /**
     * Get information about the password hash. Returns an array of the information
     * that was used to generate the password hash.
     *
     * array(
     *    'algo' => 1,
     *    'algoName' => 'bcrypt',
     *    'options' => array(
     *        'cost' => 10,
     *    ),
     * )
     *
     * @param string $hash The password hash to extract info from
     *
     * @return array The array of information about the hash.
     */
    function password_get_info($hash) {
        $return = array('algo' => 0, 'algoName' => 'unknown', 'options' => array(), );
        if (substr($hash, 0, 4) == '$2y$' && strlen($hash) == 60) {
            $return['algo'] = PASSWORD_BCRYPT;
            $return['algoName'] = 'bcrypt';
            list($cost) = sscanf($hash, "$2y$%d$");
            $return['options']['cost'] = $cost;
        }
        return $return;
    }

    /**
     * Determine if the password hash needs to be rehashed according to the options provided
     *
     * If the answer is true, after validating the password using password_verify, rehash it.
     *
     * @param string $hash    The hash to test
     * @param int    $algo    The algorithm used for new password hashes
     * @param array  $options The options array passed to password_hash
     *
     * @return boolean True if the password needs to be rehashed.
     */
    function password_needs_rehash($hash, $algo, array $options = array()) {
        $info = password_get_info($hash);
        if ($info['algo'] != $algo) {
            return true;
        }
        switch ($algo) {
            case PASSWORD_BCRYPT :
                $cost = isset($options['cost']) ? $options['cost'] : 10;
                if ($cost != $info['options']['cost']) {
                    return true;
                }
                break;
        }
        return false;
    }

    /**
     * Verify a password against a hash using a timing attack resistant approach
     *
     * @param string $password The password to verify
     * @param string $hash     The hash to verify against
     *
     * @return boolean If the password matches the hash
     */
    public function password_verify($password, $hash) {
        if (!function_exists('crypt')) {
            trigger_error("Crypt must be loaded for password_verify to function", E_USER_WARNING);
            return false;
        }
        $ret = crypt($password, $hash);
        if (!is_string($ret) || strlen($ret) != strlen($hash) || strlen($ret) <= 13) {
            return false;
        }

        $status = 0;
        for ($i = 0; $i < strlen($ret); $i++) {
            $status |= (ord($ret[$i]) ^ ord($hash[$i]));
        }

        return $status === 0;
    }

}
Димитрис Маниатис
источник
0

Я создал функцию, которую все время использую для проверки пароля и создания паролей, например, для их хранения в базе данных MySQL. Он использует случайно сгенерированную соль, что намного безопаснее, чем использование статической соли.

function secure_password($user_pwd, $multi) {

/*
    secure_password ( string $user_pwd, boolean/string $multi ) 

    *** Description: 
        This function verifies a password against a (database-) stored password's hash or
        returns $hash for a given password if $multi is set to either true or false

    *** Examples:
        // To check a password against its hash
        if(secure_password($user_password, $row['user_password'])) {
            login_function();
        } 
        // To create a password-hash
        $my_password = 'uber_sEcUrE_pass';
        $hash = secure_password($my_password, true);
        echo $hash;
*/

// Set options for encryption and build unique random hash
$crypt_options = ['cost' => 11, 'salt' => mcrypt_create_iv(22, MCRYPT_DEV_URANDOM)];
$hash = password_hash($user_pwd, PASSWORD_BCRYPT, $crypt_options);

// If $multi is not boolean check password and return validation state true/false
if($multi!==true && $multi!==false) {
    if (password_verify($user_pwd, $table_pwd = $multi)) {
        return true; // valid password
    } else {
        return false; // invalid password
    }
// If $multi is boolean return $hash
} else return $hash;

}
Геррит Фрис
источник
6
Лучше всего опустить saltпараметр, он будет автоматически сгенерирован функцией password_hash () в соответствии с лучшими практиками. Вместо этого PASSWORD_BCRYPTможно использовать PASSWORD_DEFAULTдля написания кода будущего.
martinstoeckli
Спасибо за совет. Я должен был следить за этим в документации. Это были долгие ночи.
Геррит Фрис,
1
Согласно secure.php.net/manual/en/function.password-hash.php «опция соли устарела с PHP 7.0.0. Теперь предпочтительнее просто использовать соль, которая генерируется по умолчанию».
jmng