• 首页
  • vue
  • TypeScript
  • JavaScript
  • scss
  • css3
  • html5
  • php
  • MySQL
  • redis
  • jQuery
  • crypt()

    (PHP 4, PHP 5, PHP 7)

    单向字符串散列

    说明

    crypt(string $str[,string $salt] ): string

    crypt()返回一个基于标准UNIXDES算法或系统上其他可用的替代算法的散列字符串。

    $salt参数是可选的。然而,如果没有$salt的话,crypt()创建出来的会是弱密码。 php 5.6及之后的版本会在没有它的情况下抛出一个 E_NOTICE 级别的错误。为了更好的安全性,请确保指定一个足够强度的盐值。

    password_hash()使用了一个强的哈希算法,来产生足够强的盐值,并且会自动进行合适的轮次。password_hash()是crypt()的一个简单封装,并且完全与现有的密码哈希兼容。推荐使用password_hash()。

    有些系统支持不止一种散列类型。实际上,有时候,基于 MD5 的算法被用来替代基于标准 DES 的算法。这种散列类型由盐值参数触发。在 5.3 之前,PHP 在安装时根据系统的 crypt()决定可用的算法。如果没有提供盐值,PHP 将自动生成一个 2 个字符(DES)或者 12 个字符(MD5)的盐值,这取决于 MD5 crypt()的可用性。PHP 设置了一个名为CRYPT_SALT_LENGTH的常量,用来表示可用散列允许的最长可用盐值。

    基于标准 DES 算法的crypt()在输出内容的开始位置返回两个字符的盐值。它也只使用$str的开始 8 个字符,所以更长的以相同 8 个字符开始的字符串也将生成相同的结果(当使用了相同的盐值时)。

    在 crypt()函数支持多重散列的系统上,下面的常量根据相应的类型是否可用被设置为 0 或 1:

    • CRYPT_STD_DES- 基于标准 DES 算法的散列使用"./0-9A-Za-z"字符中的两个字符作为盐值。在盐值中使用非法的字符将导致 crypt()失败。
    • CRYPT_EXT_DES- 扩展的基于 DES 算法的散列。其盐值为 9 个字符的字符串,由 1 个下划线后面跟着 4 字节循环次数和 4 字节盐值组成。它们被编码成可打印字符,每个字符 6 位,有效位最少的优先。0 到 63 被编码为"./0-9A-Za-z"。在盐值中使用非法的字符将导致 crypt()失败。
    • CRYPT_MD5- MD5 散列使用一个以$1$开始的 12 字符的字符串盐值。
    • CRYPT_BLOWFISH- Blowfish 算法使用如下盐值:“$2a$”,一个两位 cost 参数,“$”以及 64 位由“./0-9A-Za-z”中的字符组合而成的字符串。在盐值中使用此范围之外的字符将导致 crypt()返回一个空字符串。两位 cost 参数是循环次数以 2 为底的对数,它的范围是 04-31,超出这个范围将导致 crypt()失败。 PHP 5.3.7 之前只支持“$2a$”作为盐值的前缀,PHP 5.3.7 开始引入了新的前缀来修正一个在Blowfish实现上的安全风险。可以参考» this document来了解关于这个修复的更多信息。总而言之,开发者如果仅针对 PHP 5.3.7及之后版本进行开发,那应该使用“$2y$”而非“$2a$”
    • CRYPT_SHA256- SHA-256 算法使用一个以$5$开头的 16 字符字符串盐值进行散列。如果盐值字符串以“rounds=<N>$”开头,N 的数字值将被用来指定散列循环的执行次数,这点很像 Blowfish 算法的 cost 参数。默认的循环次数是 5000,最小是 1000,最大是 999,999,999。超出这个范围的 N 将会被转换为最接近的值。
    • CRYPT_SHA512- SHA-512 算法使用一个以$6$开头的 16 字符字符串盐值进行散列。如果盐值字符串以“rounds=<N>$”开头,N 的数字值将被用来指定散列循环的执行次数,这点很像 Blowfish 算法的 cost 参数。默认的循环次数是 5000,最小是 1000,最大是 999,999,999。超出这个范围的 N 将会被转换为最接近的值。
    Note:

    从 PHP 5.3.0 起,PHP 包含了它自己的实现,并将在系统缺乏相应算法支持的时候使用它自己的实现。

    参数

    $str

    待散列的字符串。

    Caution

    使用CRYPT_BLOWFISH算法将导致$str被裁剪为一个最长72个字符的字符串。

    $salt

    可选的盐值字符串。如果没有提供,算法行为将由不同的算法实现决定,并可能导致不可预料的结束。

    返回值

    返回散列后的字符串或一个少于 13 字符的字符串,从而保证在失败时与盐值区分开来。

    Warning

    当校验密码时,应该使用一个不容易被时间攻击的字符串比较函数来比较crypt()的输出与之前已知的哈希。出于这个目的,PHP5.6开始提供了hash_equals()。

    更新日志

    版本说明
    5.6.5When the failure string "*0" is given as the$salt,"*1" will now be returned for consistency with other crypt implementations. Prior to this version, PHP 5.6 would incorrectly return a DES hash.
    5.6.0Raise E_NOTICE security warning if$saltis omitted.
    5.5.21When the failure string "*0" is given as the$salt,"*1" will now be returned for consistency with other crypt implementations. Prior to this version, PHP 5.5 (and earlier branches) would incorrectly return a DES hash.
    5.3.7Added$2x$and$2y$Blowfish modes to deal with potential high-bit attacks.
    5.3.2基于 Ulrich Drepper 的»实现,新增基于 SHA-256 算法和 SHA-512 算法的 crypt。
    5.3.2修正了 Blowfish 算法由于非法循环导致的问题,返回“失败”字符串(“*0”或“*1”)而不是转而使用 DES 算法。
    5.3.0PHP 现在包含了它自己的 MD5 Crypt 实现,包括标准 DES 算法,扩展的 DES 算法以及 Blowfish 算法。如果系统缺乏相应的实现,那么 PHP 将使用它自己的实现。

    范例

    Example #1crypt()范例

    <?php
    $hashed_password = crypt('mypassword'); // 自动生成盐值
    /* 你应当使用 crypt() 得到的完整结果作为盐值进行密码校验,以此来避免使用不同散列算法导致的问题。(如上所述,基于标准 DES 算法的密码散列使用 2 字符盐值,但是基于 MD5 算法的散列使用 12 个字符盐值。)*/
    if (hash_equals($hashed_password, crypt($user_input, $hashed_password))) {
       echo "Password verified!";
    }
    ?>

    利用 htpasswd 进行crypt()加密

    <?php
    // 设置密码
    $password = 'mypassword';
    // 获取散列值,使用自动盐值
    $hash = crypt($password);
    ?>

    以不同散列类型使用crypt()

    <?php
    if (CRYPT_STD_DES == 1) {
        echo 'Standard DES: ' . crypt('rasmuslerdorf', 'rl') . "\n";
    }
    if (CRYPT_EXT_DES == 1) {
        echo 'Extended DES: ' . crypt('rasmuslerdorf', '_J9..rasm') . "\n";
    }
    if (CRYPT_MD5 == 1) {
        echo 'MD5:          ' . crypt('rasmuslerdorf', '$1$rasmusle$') . "\n";
    }
    if (CRYPT_BLOWFISH == 1) {
        echo 'Blowfish:     ' . crypt('rasmuslerdorf', '$2a$07$usesomesillystringforsalt$') . "\n";
    }
    if (CRYPT_SHA256 == 1) {
        echo 'SHA-256:      ' . crypt('rasmuslerdorf', '$5$rounds=5000$usesomesillystringforsalt$') . "\n";
    }
    if (CRYPT_SHA512 == 1) {
        echo 'SHA-512:      ' . crypt('rasmuslerdorf', '$6$rounds=5000$usesomesillystringforsalt$') . "\n";
    }
    ?>

    以上例程的输出类似于:

    Standard DES: rl.3StKT.4T8M
    Extended DES: _J9..rasmBYk8r9AiWNc
    MD5:          $1$rasmusle$rISCgZzpwk3UhDidwXvin0
    Blowfish:     $2a$07$usesomesillystringfore2uDLvp1Ii2e./U9C8sBjqp8I90dH6hi
    SHA-256:      $5$rounds=5000$usesomesillystri$KqJWpanXZHKq2BOB43TSaYhEWsQ1Lr5QNyPCDH/Tp.6
    SHA-512:      $6$rounds=5000$usesomesillystri$D4IrlXatmP7rx3P3InaxBeoomnAihCKRVQP22JZ6EY47Wc6BkroIuUUBOov1i.S5KPgErtP/EN5mcO.ChWQW21
    

    注释

    Note:由于crypt()使用的是单向算法,因此不存在 decrypt 函数。

    参见

    • hash_equals()可防止时序攻击的字符串比较
    • password_hash()创建密码的散列(hash)
    • md5()计算字符串的 MD5 散列值
    • Mcrypt扩展
    • 更多关于 crypt 函数的信息,请阅读 Unix man 页面
    The #2 comment on this comments page (as of Feb 2015) is 9 years old and recommends phpass. I have independently security audited this product and, while it continues to be recommended for password security, it is actually insecure and should NOT be used. It hasn't seen any updates in years (still at v0.3) and there are more recent alternatives such as using the newer built-in PHP password_hash() function that are much better. Everyone, please take a few moments to confirm what I'm saying is accurate (i.e. review the phpass code for yourself) and then click the down arrow to sink the phpass comment to the bottom. You'll be increasing security across the Internet by doing so.
    For those who want details: md5() with microtime() are a fallback position within the source code of phpass. Instead of terminating, it continues to execute code. The author's intentions of trying to work everywhere are admirable but, when it comes to application security, that stance actually backfires. The only correct answer in a security context is to terminate the application rather than fallback to a weak position that can potentially be exploited (usually by forcing that weaker position to happen).
    As I understand it, blowfish is generally seen a secure hashing algorithm, even for enterprise use (correct me if I'm wrong). Because of this, I created functions to create and check secure password hashes using this algorithm, and using the (also deemed cryptographically secure) openssl_random_pseudo_bytes function to generate the salt.
    <?php
    /*
     * Generate a secure hash for a given password. The cost is passed
     * to the blowfish algorithm. Check the PHP manual page for crypt to
     * find more information about this setting.
     */
    function generate_hash($password, $cost=11){
        /* To generate the salt, first generate enough random bytes. Because
         * base64 returns one character for each 6 bits, the we should generate
         * at least 22*6/8=16.5 bytes, so we generate 17. Then we get the first
         * 22 base64 characters
         */
        $salt=substr(base64_encode(openssl_random_pseudo_bytes(17)),0,22);
        /* As blowfish takes a salt with the alphabet ./A-Za-z0-9 we have to
         * replace any '+' in the base64 string with '.'. We don't have to do
         * anything about the '=', as this only occurs when the b64 string is
         * padded, which is always after the first 22 characters.
         */
        $salt=str_replace("+",".",$salt);
        /* Next, create a string that will be passed to crypt, containing all
         * of the settings, separated by dollar signs
         */
        $param='$'.implode('$',array(
            "2y", //select the most secure version of blowfish (>=PHP 5.3.7)
            str_pad($cost,2,"0",STR_PAD_LEFT), //add the cost in two digits
            $salt //add the salt
        ));
        
        //now do the actual hashing
        return crypt($password,$param);
    }
     
    /*
     * Check the password against a hash generated by the generate_hash
     * function.
     */
    function validate_pw($password, $hash){
        /* Regenerating the with an available hash as the options parameter should
         * produce the same hash if the same password is passed.
         */
        return crypt($password, $hash)==$hash;
    }
    ?>
    To generate salt use mcrypt_create_iv() not mt_rand() because no matter how many times you call mt_rand() it will only have at most 32 bits of entropy. Which you will start seeing salt collisions after about 2^16 users. mt_rand() is seeded poorly so it should happen sooner.
    For bcrypt this will actually generate a 128 bit salt:
    <?php $salt = strtr(base64_encode(mcrypt_create_iv(16, MCRYPT_DEV_URANDOM)), '+', '.'); ?>
    *** Bike shed ***
    The last character in the 22 character salt is 2 bits.
    base64_encode() will have these four character "AQgw"
    bcrypt will have these four character ".Oeu"
    You don't need to do a full translate because they "round" to different characters:
    echo crypt('', '$2y$05$.....................A') . "\n";
    echo crypt('', '$2y$05$.....................Q') . "\n";
    echo crypt('', '$2y$05$.....................g') . "\n";
    echo crypt('', '$2y$05$.....................w') . "\n";
    $2y$05$......................J2ihDv8vVf7QZ9BsaRrKyqs2tkn55Yq
    $2y$05$.....................O/jw2XygQa2.LrIT7CFCBQowLowDP6Y.
    $2y$05$.....................eDOx4wMcy7WU.kE21W6nJfdMimsBE3V6
    $2y$05$.....................uMMcgjnOELIa6oydRivPkiMrBG8.aFp.
    Here is an expression to generate pseudorandom salt for the CRYPT_BLOWFISH hash type:
    <?php $salt = substr(str_replace('+', '.', base64_encode(pack('N4', mt_rand(), mt_rand(), mt_rand(), mt_rand()))), 0, 22); ?>
    It is intended for use on systems where mt_getrandmax() == 2147483647.
    The salt created will be 128 bits in length, padded to 132 bits and then expressed in 22 base64 characters. (CRYPT_BLOWFISH only uses 128 bits for the salt, even though there are 132 bits in 22 base64 characters. If you examine the CRYPT_BLOWFISH input and output, you can see that it ignores the last four bits on input, and sets them to zero on output.)
    Note that the high-order bits of the four 32-bit dwords returned by mt_rand() will always be zero (since mt_getrandmax == 2^31), so only 124 of the 128 bits will be pseudorandom. I found that acceptable for my application.
    The crypt() function cant handle plus signs correctly. So if for example you are using crypt in a login function, use urlencode on the password first to make sure that the login procedure can handle any character:
    <?php
    $user_input = '12+#æ345';
    $pass = urlencode($user_input));
    $pass_crypt = crypt($pass);
    if ($pass_crypt == crypt($pass, $pass_crypt)) {
     echo "Success! Valid password";
    } else {
     echo "Invalid password";
    } 
    ?>
    While the documentation says that crypt will fail for DES if the salt is invalid, this turns out to not be the case.
    The crypt function will accept any string of two characters or more for DES as long as it doesn't match the pattern for any other hashing schema. The remaining characters will be ignored.
    steve at tobtu dot com was right 4 years ago, but now mcrypt_create_iv() (and bcrypt in general) is deprecated!
    Use random_bytes() instead:
    <?php
    $salt = base64_encode(random_bytes(16));
    If you're stuck with CRYPT_EXT_DES, then you'll want to pick a number of iterations: the 2nd-5th characters of the "salt".
    My experimentation suggests that the 5th character is the most significant. A '.' is a zero and 'Z' is the highest value. Using all dots will create an error: all passwords will be encrypted to the same value.
    Here are some encryption timings (in seconds) that I obtained, with five different iteration counts over the same salt, and the same password, on a quad core 2.66GHz Intel Xeon machine.
    _1111 time: 0.15666794776917
    _J9.Z time: 1.8860530853271
    _J9.. time: 0.00015401840209961
    _...Z time: 1.9095730781555
    _ZZZZ time: 1.9124970436096
    _...A time: 0.61211705207825
    I think a half a second is reasonable for an application, but for the back end authentication? I'm not so sure: there's a significant risk of overloading the back end if we're getting lots of authentication requests.

    上篇:crc32()

    下篇:echo()