2008-04-26 06:31:52 +00:00
< ? php
/**
* This class represents a validator for member passwords .
*
* < code >
* $pwdVal = new PasswordValidator ();
* $pwdValidator -> minLength ( 7 );
* $pwdValidator -> checkHistoricalPasswords ( 6 );
* $pwdValidator -> characterStrength ( 'lowercase' , 'uppercase' , 'digits' , 'punctuation' );
*
* Member :: set_password_validator ( $pwdValidator );
* </ code >
2008-06-15 13:33:53 +00:00
*
* @ package sapphire
* @ subpackage security
2008-04-26 06:31:52 +00:00
*/
class PasswordValidator extends Object {
static $character_strength_tests = array (
'lowercase' => '/[a-z]/' ,
'uppercase' => '/[A-Z]/' ,
'digits' => '/[0-9]/' ,
'punctuation' => '/[^A-Za-z0-9]/' ,
);
protected $minLength , $minScore , $testNames , $historicalPasswordCount ;
/**
* Minimum password length
*/
function minLength ( $minLength ) {
$this -> minLength = $minLength ;
}
/**
* Check the character strength of the password .
*
* Eg : $this -> characterStrength ( 3 , array ( " lowercase " , " uppercase " , " digits " , " punctuation " ))
*
* @ param $minScore The minimum number of character tests that must pass
* @ param $testNames The names of the tests to perform
*/
function characterStrength ( $minScore , $testNames ) {
$this -> minScore = $minScore ;
$this -> testNames = $testNames ;
}
/**
* Check a number of previous passwords that the user has used , and don ' t let them change to that .
*/
function checkHistoricalPasswords ( $count ) {
$this -> historicalPasswordCount = $count ;
}
2010-06-18 02:59:43 +00:00
/**
* @ param String $password
* @ param Member $member
* @ return ValidationResult
*/
2008-04-26 06:31:52 +00:00
function validate ( $password , $member ) {
$valid = new ValidationResult ();
if ( $this -> minLength ) {
2010-06-18 02:59:43 +00:00
if ( strlen ( $password ) < $this -> minLength ) $valid -> error ( sprintf ( " Password is too short, it must be %s or more characters long. " , $this -> minLength ), " TOO_SHORT " );
2008-04-26 06:31:52 +00:00
}
if ( $this -> minScore ) {
$score = 0 ;
$missedTests = array ();
foreach ( $this -> testNames as $name ) {
if ( preg_match ( self :: $character_strength_tests [ $name ], $password )) $score ++ ;
else $missedTests [] = $name ;
}
if ( $score < $this -> minScore ) {
$valid -> error ( " You need to increase the strength of your passwords by adding some of the following characters: " . implode ( " , " , $missedTests ), " LOW_CHARACTER_STRENGTH " );
}
}
if ( $this -> historicalPasswordCount ) {
2008-11-24 09:31:14 +00:00
$previousPasswords = DataObject :: get ( " MemberPassword " , " \" MemberID \" = $member->ID " , " \" Created \" DESC, \" ID \" Desc " , " " , $this -> historicalPasswordCount );
2008-04-26 06:31:52 +00:00
if ( $previousPasswords ) foreach ( $previousPasswords as $previousPasswords ) {
if ( $previousPasswords -> checkPassword ( $password )) {
$valid -> error ( " You've already used that password in the past, please choose a new password " , " PREVIOUS_PASSWORD " );
break ;
}
}
}
return $valid ;
}
}