File indexing completed on 2025-01-19 05:20:55
0001 <?php 0002 /** 0003 * Zend Framework 0004 * 0005 * LICENSE 0006 * 0007 * This source file is subject to the new BSD license that is bundled 0008 * with this package in the file LICENSE.txt. 0009 * It is also available through the world-wide-web at this URL: 0010 * http://framework.zend.com/license/new-bsd 0011 * If you did not receive a copy of the license and are unable to 0012 * obtain it through the world-wide-web, please send an email 0013 * to license@zend.com so we can send you a copy immediately. 0014 * 0015 * @category Zend 0016 * @package Zend_Auth 0017 * @subpackage Adapter 0018 * @copyright Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com) 0019 * @license http://framework.zend.com/license/new-bsd New BSD License 0020 * @version $Id$ 0021 */ 0022 0023 0024 /** 0025 * @see Zend_Auth_Adapter_Interface 0026 */ 0027 // require_once 'Zend/Auth/Adapter/Interface.php'; 0028 0029 0030 /** 0031 * @category Zend 0032 * @package Zend_Auth 0033 * @subpackage Adapter 0034 * @copyright Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com) 0035 * @license http://framework.zend.com/license/new-bsd New BSD License 0036 */ 0037 class Zend_Auth_Adapter_Digest implements Zend_Auth_Adapter_Interface 0038 { 0039 /** 0040 * Filename against which authentication queries are performed 0041 * 0042 * @var string 0043 */ 0044 protected $_filename; 0045 0046 /** 0047 * Digest authentication realm 0048 * 0049 * @var string 0050 */ 0051 protected $_realm; 0052 0053 /** 0054 * Digest authentication user 0055 * 0056 * @var string 0057 */ 0058 protected $_username; 0059 0060 /** 0061 * Password for the user of the realm 0062 * 0063 * @var string 0064 */ 0065 protected $_password; 0066 0067 /** 0068 * Sets adapter options 0069 * 0070 * @param mixed $filename 0071 * @param mixed $realm 0072 * @param mixed $username 0073 * @param mixed $password 0074 */ 0075 public function __construct($filename = null, $realm = null, $username = null, $password = null) 0076 { 0077 $options = array('filename', 'realm', 'username', 'password'); 0078 foreach ($options as $option) { 0079 if (null !== $$option) { 0080 $methodName = 'set' . ucfirst($option); 0081 $this->$methodName($$option); 0082 } 0083 } 0084 } 0085 0086 /** 0087 * Returns the filename option value or null if it has not yet been set 0088 * 0089 * @return string|null 0090 */ 0091 public function getFilename() 0092 { 0093 return $this->_filename; 0094 } 0095 0096 /** 0097 * Sets the filename option value 0098 * 0099 * @param mixed $filename 0100 * @return Zend_Auth_Adapter_Digest Provides a fluent interface 0101 */ 0102 public function setFilename($filename) 0103 { 0104 $this->_filename = (string) $filename; 0105 return $this; 0106 } 0107 0108 /** 0109 * Returns the realm option value or null if it has not yet been set 0110 * 0111 * @return string|null 0112 */ 0113 public function getRealm() 0114 { 0115 return $this->_realm; 0116 } 0117 0118 /** 0119 * Sets the realm option value 0120 * 0121 * @param mixed $realm 0122 * @return Zend_Auth_Adapter_Digest Provides a fluent interface 0123 */ 0124 public function setRealm($realm) 0125 { 0126 $this->_realm = (string) $realm; 0127 return $this; 0128 } 0129 0130 /** 0131 * Returns the username option value or null if it has not yet been set 0132 * 0133 * @return string|null 0134 */ 0135 public function getUsername() 0136 { 0137 return $this->_username; 0138 } 0139 0140 /** 0141 * Sets the username option value 0142 * 0143 * @param mixed $username 0144 * @return Zend_Auth_Adapter_Digest Provides a fluent interface 0145 */ 0146 public function setUsername($username) 0147 { 0148 $this->_username = (string) $username; 0149 return $this; 0150 } 0151 0152 /** 0153 * Returns the password option value or null if it has not yet been set 0154 * 0155 * @return string|null 0156 */ 0157 public function getPassword() 0158 { 0159 return $this->_password; 0160 } 0161 0162 /** 0163 * Sets the password option value 0164 * 0165 * @param mixed $password 0166 * @return Zend_Auth_Adapter_Digest Provides a fluent interface 0167 */ 0168 public function setPassword($password) 0169 { 0170 $this->_password = (string) $password; 0171 return $this; 0172 } 0173 0174 /** 0175 * Defined by Zend_Auth_Adapter_Interface 0176 * 0177 * @throws Zend_Auth_Adapter_Exception 0178 * @return Zend_Auth_Result 0179 */ 0180 public function authenticate() 0181 { 0182 $optionsRequired = array('filename', 'realm', 'username', 'password'); 0183 foreach ($optionsRequired as $optionRequired) { 0184 if (null === $this->{"_$optionRequired"}) { 0185 /** 0186 * @see Zend_Auth_Adapter_Exception 0187 */ 0188 // require_once 'Zend/Auth/Adapter/Exception.php'; 0189 throw new Zend_Auth_Adapter_Exception("Option '$optionRequired' must be set before authentication"); 0190 } 0191 } 0192 0193 if (false === ($fileHandle = @fopen($this->_filename, 'r'))) { 0194 /** 0195 * @see Zend_Auth_Adapter_Exception 0196 */ 0197 // require_once 'Zend/Auth/Adapter/Exception.php'; 0198 throw new Zend_Auth_Adapter_Exception("Cannot open '$this->_filename' for reading"); 0199 } 0200 0201 $id = "$this->_username:$this->_realm"; 0202 $idLength = strlen($id); 0203 0204 $result = array( 0205 'code' => Zend_Auth_Result::FAILURE, 0206 'identity' => array( 0207 'realm' => $this->_realm, 0208 'username' => $this->_username, 0209 ), 0210 'messages' => array() 0211 ); 0212 0213 while ($line = trim(fgets($fileHandle))) { 0214 if (substr($line, 0, $idLength) === $id) { 0215 if ($this->_secureStringCompare(substr($line, -32), md5("$this->_username:$this->_realm:$this->_password"))) { 0216 $result['code'] = Zend_Auth_Result::SUCCESS; 0217 } else { 0218 $result['code'] = Zend_Auth_Result::FAILURE_CREDENTIAL_INVALID; 0219 $result['messages'][] = 'Password incorrect'; 0220 } 0221 return new Zend_Auth_Result($result['code'], $result['identity'], $result['messages']); 0222 } 0223 } 0224 0225 $result['code'] = Zend_Auth_Result::FAILURE_IDENTITY_NOT_FOUND; 0226 $result['messages'][] = "Username '$this->_username' and realm '$this->_realm' combination not found"; 0227 return new Zend_Auth_Result($result['code'], $result['identity'], $result['messages']); 0228 } 0229 0230 /** 0231 * Securely compare two strings for equality while avoided C level memcmp() 0232 * optimisations capable of leaking timing information useful to an attacker 0233 * attempting to iteratively guess the unknown string (e.g. password) being 0234 * compared against. 0235 * 0236 * @param string $a 0237 * @param string $b 0238 * @return bool 0239 */ 0240 protected function _secureStringCompare($a, $b) 0241 { 0242 if (strlen($a) !== strlen($b)) { 0243 return false; 0244 } 0245 $result = 0; 0246 for ($i = 0; $i < strlen($a); $i++) { 0247 $result |= ord($a[$i]) ^ ord($b[$i]); 0248 } 0249 return $result == 0; 0250 } 0251 }