downloads | documentation | faq | getting help | mailing lists | licenses | wiki | reporting bugs | php.net sites | links | conferences | my php.net

search for in the

echo> <crc32
[edit] Last updated: Fri, 18 Sep 2009

view this page in

crypt

(PHP 4, PHP 5)

cryptЕднопосочно криптиране на низ (хеширане)

Описание

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

crypt() връща низ, криптиран на базата на стандарния Unix DES алгоритъм за криптиране или на базата на други алгоритми достъпни в системата.

Някои операционни системи поддържат повече от един вид криптиране. Всъщност, понякога стандартното криптиране базирано на DES алгоритъма се заменя с криптиране базирано на MD5 алгоритъма. Типът на криптиране зависи от salt параметъра. По време на инсталацията, PHP определя възможностите на функцията crypt и приема ключове и за други типове криптирания. Ако не е зададен ключ, PHP ще генерира стандартен двузнаков salt по подразбиране, освен ако текущият тип на криптиране на системата е MD5. В този случай ще се генерира случаен MD5-съвместим salt. В PHP е установена константата CRYPT_SALT_LENGTH, която указва дали обикновен двузнаков salt може да се използва във вашата система или е приложим по-дългият дванадесет знаков salt.

Ако използвате предоставения salt, трябва да знаете, че той се генерира само веднъж. Ако извиквате тази функция многократно, това може да повлияе на изхода и на сигурността. Функцията crypt(), която използва стандартно DES криптиране връща salt-а като първите два символа от изхода. Също така, използва само първите осем знака от параметъра str , така че, по-дълги низове, започващи със същите осем знака ще генерират същият резултат (ако се използва същият salt).

В системи, при които функцията crypt() поддържа множество типове криптиране, следните константи са установени в 0 или 1, в зависимост дали дадения тип е наличен:

  • CRYPT_STD_DES - Стандартно криптиране базирано на DES със двузнаков salt
  • CRYPT_EXT_DES - Разширено криптиране базирано на DES със девет знаков salt
  • CRYPT_MD5 - MD5 криптиране със дванадесет знаков salt започващ с $1$
  • CRYPT_BLOWFISH - Blowfish криптиране със шестнадесет знаков salt започващ с $2$ или $2a$

Параметри

str

Низът, който ще се криптира.

salt

Незадължителен salt низ, на базата на който се извършва криптирането. Ако не е зададен, такъв ще бъде генериран по случаен начин от PHP при всяко извикване на тази функция.

Важно е да се знае, че ако е зададен salt низа, то ключа се генерира само веднъж. При повторното извикване на функцията това може да повлияе на резултата от изпълнението и на сигурността.

Връщани стойности

Връща криптиран низ.

Примери

Example #1 Примери за crypt()

<?php
$password 
crypt('mypassword'); // нека salt параметъра се генерира автоматично

/* Трябва да предадете целия резултат от crypt() като salt при сравняване
   на парола, за да избегнете проблеми при употребата на различни алгоритми
   за хеширане. (Както е казано по-горе, стандартното базирано на DES хеширане
   използва двузнаков salt, но базираното на MD5 алгоритъма хеширане използва 12.) */
if (crypt($user_input$password) == $password) {
   echo 
"Password verified!";
}
?>

Example #2 Употреба на crypt() с htpasswd

<?php
// Установяване на паролата
$password 'mypassword';

// Връща хеш стойността, оставяйки salt параметъра да бъде автоматично генериран
$hash crypt($password);
?>

Example #3 Употреба на 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$rasmuslerd...........$') . "\n";
}
?>

Примерът по-горе ще изведе нещо подобно на:

Standard DES: rl.3StKT.4T8M
Extended DES: _J9..rasmBYk8r9AiWNc
MD5:          $1$rasmusle$rISCgZzpwk3UhDidwXvin0
Blowfish:     $2a$07$rasmuslerd............nIdrcHdxcUxWomQX9j6kvERCFjTg7Ra

Бележки

Забележка: Функция за декриптиране не съществува, тъй като crypt() използва еднопосочен алгоритъм.

Вж. също

  • md5() - Изчислява md5 хеш на низ
  • Разширението Mcrypt
  • Страницата от ръководството на Unix за функцията crypt



echo> <crc32
[edit] Last updated: Fri, 18 Sep 2009
 
add a note add a note User Contributed Notes crypt
Matteo 18-May-2012 10:39
Password hashing should be done only with crypt and NEVER with SHA* and MD5 or hash(). The fundamental reason is that crypt is designed to be SLOW which is a VERY good thing for password hashing.

It also automatically generate a salt every time which makes pre-computed tables to "decrypt" passwords useless (the generated salt is stored in the returned string for convenience).
chris at ocportal dot com 29-Mar-2012 09:49
If you need to support older versions of PHP be aware that constants such as CRYPT_BLOWFISH may not actually be defined. So rather than following Example #3, you need a defined('CRYPT_...') call there before checking the constant's value.
Anonymous 19-Feb-2012 09:21
This function is not binary safe. Any binary string containing a NULL byte (chr(0)) will not produce a valid hash.
harry at simans dot net 27-Sep-2011 04:34
I made a nice little wrapper function for crypt():

<?php
function hasher($info, $encdata = false)
{
 
$strength = "08";
 
//if encrypted data is passed, check it against input ($info)
 
if ($encdata) {
    if (
substr($encdata, 0, 60) == crypt($info, "$2a$".$strength."$".substr($encdata, 60))) {
      return
true;
    }
    else {
      return
false;
    }
  }
  else {
 
//make a salt and hash it with input, and add salt to end
 
$salt = "";
  for (
$i = 0; $i < 22; $i++) {
   
$salt .= substr("./ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789", mt_rand(0, 63), 1);
  }
 
//return 82 char string (60 char hash & 22 char salt)
return crypt($info, "$2a$".$strength."$".$salt).$salt;
}
}
?>

This wrapper will accept a string as input and hash it, and output the hash result of the string and salt together, plus the salt added on the end. You can then store that output in a db, and pass it on to the function as the 2nd parameter when you go to verify it, along with the user input or whatever as the first.

Examples:

<?php
$hash
= hasher($userinput);
if (
$hash == hasher($userinput, $hash) {//authed}
?>

Neat huh?
kaminski at istori dot com 05-Feb-2011 03:43
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.
php at nospam dot nowhere dot com 26-Aug-2009 07:21
The makesalt() function code below when used to create an MD5 salt, produces a salt with characters not typically in a salt used by operating system crypt functions.  Some of these characters may have unintended side effects depending on how they are used - including the following: @ ` ~ \ | {}.

I am using the following to create MD5-Crypt hashes, (yes, I am assuming CRYPT_MD5 support is present).

<?php
function md5crypt($password){
   
// create a salt that ensures crypt creates an md5 hash
   
$base64_alphabet='ABCDEFGHIJKLMNOPQRSTUVWXYZ'
                   
.'abcdefghijklmnopqrstuvwxyz0123456789+/';
   
$salt='$1$';
    for(
$i=0; $i<9; $i++){
       
$salt.=$base64_alphabet[rand(0,63)];
    }
   
// return the crypt md5 password
   
return crypt($password,$salt.'$');
}
?>
Jou 06-Apr-2007 01:52
I found out that you can use php:s crypt function to change  the user/root password in Linux distributions (at least in Slackware).

You just have to change the encrypted password for the user in the /etc/shadow file with the output from crypt("newpassword");
mikey_nich (at) hotmáil . com 04-Mar-2007 07:47
Are you using Apache2 on f.i. WinXP and want to create .htpasswd files via php? Then you need to use the APR1-MD5 encryption method. Here is a function for that:

<?php

function crypt_apr1_md5($plainpasswd) {
   
$salt = substr(str_shuffle("abcdefghijklmnopqrstuvwxyz0123456789"), 0, 8);
   
$len = strlen($plainpasswd);
   
$text = $plainpasswd.'$apr1$'.$salt;
   
$bin = pack("H32", md5($plainpasswd.$salt.$plainpasswd));
    for(
$i = $len; $i > 0; $i -= 16) { $text .= substr($bin, 0, min(16, $i)); }
    for(
$i = $len; $i > 0; $i >>= 1) { $text .= ($i & 1) ? chr(0) : $plainpasswd{0}; }
   
$bin = pack("H32", md5($text));
    for(
$i = 0; $i < 1000; $i++) {
       
$new = ($i & 1) ? $plainpasswd : $bin;
        if (
$i % 3) $new .= $salt;
        if (
$i % 7) $new .= $plainpasswd;
       
$new .= ($i & 1) ? $bin : $plainpasswd;
       
$bin = pack("H32", md5($new));
    }
    for (
$i = 0; $i < 5; $i++) {
       
$k = $i + 6;
       
$j = $i + 12;
        if (
$j == 16) $j = 5;
       
$tmp = $bin[$i].$bin[$k].$bin[$j].$tmp;
    }
   
$tmp = chr(0).chr(0).$bin[11].$tmp;
   
$tmp = strtr(strrev(substr(base64_encode($tmp), 2)),
   
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",
   
"./0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz");
    return
"$"."apr1"."$".$salt."$".$tmp;
}

?>
picolobo at pilab dot dyndns dot org 21-Sep-2006 01:49
I had problems with ENCRYPT MySQL function when i tried to compare with the encrypted password (with ENCRYPT).

Another solution i read from "UNIX Advanced programming" where i found about the UNIX system call "crypt()":

Password="tB" //The two first letters of encrypted password

SELECT password from users where Password=ENCRYPT('".$_POST['password']."',Password)

mysql> select password from users where password=encrypt('pasword','tB');
+---------------+
| password      |
+---------------+
| tBY8OVuabSiTU |
+---------------+
1 row in set (0.01 sec)

Bye.

> topace at lightbox dot org
> 22-Sep-2005 06:34
>
> To authenticate against a stored crypt in MySQL, simply use:
>
> SELECT ................
>           AND Password=ENCRYPT('".$_POST['password']."',Password)
solar at openwall dot com 23-Dec-2005 11:20
With different password hashing methods supported on different systems and with the need to generate salts with your own PHP code in order to use the more advanced / more secure methods, it takes special knowledge to use crypt() optimally, producing strong password hashes.  Other message digest / hashing functions supported by PHP, such as md5() and sha1(), are really no good for password hashing if used naively, resulting in hashes which may be brute-forced at rates much higher than those possible for hashes produced by crypt().

I have implemented a PHP password hashing framework (in PHP, tested with all of PHP 3, 4, and 5) which hides the complexity from your PHP applications (no need for you to worry about salts, etc.), yet does things in almost the best way possible given the constraints of the available functions.  The homepage for the framework is:

http://www.openwall.com/phpass/

I have placed this code in the public domain, so there are no copyrights or licensing restrictions to worry about.

P.S. I have 10 years of experience in password (in)security and I've developed several other password security tools and libraries.  So most people can feel confident they're getting this done better by using my framework than they could have done it on their own.
hotdog (at) gmx (dot) net 16-Nov-2005 07:34
WRONG:

$mypassword = "toto";
$smd5_pass = "{SMD5}......." // in openldap

if (preg_match ("/{SMD5}/i", $smd5_pass))
 {
  $encrypted = substr($md5_pass, 6);
  $hash = base64_decode($encrypted);
  $salt = substr($hash,16);
  $mhashed =  mhash(MHASH_MD5, $mypassword . $salt) ;
  $without_salt = explode($salt,$hash_hex);
   if ($without_salt[0] == $mhashed) {
    echo "Password verified <br>";
    } else {
    echo "Password Not verified<br>";
    }
 }

$without_salt = explode($salt,$hash_hex); should be $without_salt = explode($salt,$hash);

RIGHT:

$mypassword = "toto";
$smd5_pass = "{SMD5}......." // in openldap

if (preg_match ("/{SMD5}/i", $smd5_pass))
 {
  $encrypted = substr($md5_pass, 6);
  $hash = base64_decode($encrypted);
  $salt = substr($hash,16);
  $mhashed =  mhash(MHASH_MD5, $mypassword . $salt) ;
  $without_salt = explode($salt,$hash);
   if ($without_salt[0] == $mhashed) {
    echo "Password verified <br>";
    } else {
    echo "Password Not verified<br>";
    }
 }
Vlad Alexa Mancini mancin at nextcode dot org 15-May-2005 01:57
cleaner version of shadow() and with more ascii chars

<?php

function shadow ($input){
         for (
$n = 0; $n < 9; $n++){
             
$s .= chr(rand(64,126));
         }
        
$seed "$1$".$s."$";
        
$return = crypt($input,$seed);
    return
$return;
}

>
thorhajo at gmail dot com 02-Sep-2004 03:34
Here's a little function I wrote to generate MD5 password hashes in the format they're found in /etc/shadow:

function shadow($password)
{
  $hash = '';
  for($i=0;$i<8;$i++)
  {
    $j = mt_rand(0,53);
    if($j<26)$hash .= chr(rand(65,90));
    else if($j<52)$hash .= chr(rand(97,122));
    else if($j<53)$hash .= '.';
    else $hash .= '/';
  }
  return crypt($password,'$1$'.$hash.'$');
}

I've written this so that each character in the a-zA-Z./ set has a 1/54 of a chance of being selected (26 + 26 + 2 = 54), thus being statistically even.
aidan at php dot net 05-Jul-2004 07:52
Text_Password allows one to create pronounceable and unpronounceable passwords.

http://pear.php.net/package/text_password

 
show source | credits | stats | sitemap | contact | advertising | mirror sites