<?php
/**
 * Zend Framework (http://framework.zend.com/)
 *
 * @link      http://github.com/zendframework/zf2 for the canonical source repository
 * @copyright Copyright (c) 2005-2013 Zend Technologies USA Inc. (http://www.zend.com)
 * @license   http://framework.zend.com/license/new-bsd New BSD License
 */

namespace Zend\Authentication\Adapter;

use 
Zend\Authentication\Result as AuthenticationResult;
use 
Zend\Stdlib\ErrorHandler;

class 
Digest extends AbstractAdapter
{
    
/**
     * Filename against which authentication queries are performed
     *
     * @var string
     */
    
protected $filename;

    
/**
     * Digest authentication realm
     *
     * @var string
     */
    
protected $realm;

    
/**
     * Sets adapter options
     *
     * @param  mixed $filename
     * @param  mixed $realm
     * @param  mixed $identity
     * @param  mixed $credential
     */
    
public function __construct($filename null$realm null$identity null$credential null)
    {
        if (
$filename !== null) {
            
$this->setFilename($filename);
        }
        if (
$realm !== null) {
            
$this->setRealm($realm);
        }
        if (
$identity !== null) {
            
$this->setIdentity($identity);
        }
        if (
$credential !== null) {
            
$this->setCredential($credential);
        }
    }

    
/**
     * Returns the filename option value or null if it has not yet been set
     *
     * @return string|null
     */
    
public function getFilename()
    {
        return 
$this->filename;
    }

    
/**
     * Sets the filename option value
     *
     * @param  mixed $filename
     * @return Digest Provides a fluent interface
     */
    
public function setFilename($filename)
    {
        
$this->filename = (string) $filename;
        return 
$this;
    }

    
/**
     * Returns the realm option value or null if it has not yet been set
     *
     * @return string|null
     */
    
public function getRealm()
    {
        return 
$this->realm;
    }

    
/**
     * Sets the realm option value
     *
     * @param  mixed $realm
     * @return Digest Provides a fluent interface
     */
    
public function setRealm($realm)
    {
        
$this->realm = (string) $realm;
        return 
$this;
    }

    
/**
     * Returns the username option value or null if it has not yet been set
     *
     * @return string|null
     */
    
public function getUsername()
    {
        return 
$this->getIdentity();
    }

    
/**
     * Sets the username option value
     *
     * @param  mixed $username
     * @return Digest Provides a fluent interface
     */
    
public function setUsername($username)
    {
        return 
$this->setIdentity($username);
    }

    
/**
     * Returns the password option value or null if it has not yet been set
     *
     * @return string|null
     */
    
public function getPassword()
    {
        return 
$this->getCredential();
    }

    
/**
     * Sets the password option value
     *
     * @param  mixed $password
     * @return Digest Provides a fluent interface
     */
    
public function setPassword($password)
    {
        return 
$this->setCredential($password);
    }

    
/**
     * Defined by Zend\Authentication\Adapter\AdapterInterface
     *
     * @throws Exception\ExceptionInterface
     * @return AuthenticationResult
     */
    
public function authenticate()
    {
        
$optionsRequired = array('filename''realm''identity''credential');
        foreach (
$optionsRequired as $optionRequired) {
            if (
null === $this->$optionRequired) {
                throw new 
Exception\RuntimeException("Option '$optionRequired' must be set before authentication");
            }
        }

        
ErrorHandler::start(E_WARNING);
        
$fileHandle fopen($this->filename'r');
        
$error      ErrorHandler::stop();
        if (
false === $fileHandle) {
            throw new 
Exception\UnexpectedValueException("Cannot open '$this->filename' for reading"0$error);
        }

        
$id       "$this->identity:$this->realm";
        
$idLength strlen($id);

        
$result = array(
            
'code'  => AuthenticationResult::FAILURE,
            
'identity' => array(
                
'realm'    => $this->realm,
                
'username' => $this->identity,
            ),
            
'messages' => array()
        );

        while ((
$line fgets($fileHandle)) !== false) {
            
$line trim($line);
            if (empty(
$line)) {
                break;
            }
            if (
substr($line0$idLength) === $id) {
                if (
$this->_secureStringCompare(substr($line, -32), md5("$this->identity:$this->realm:$this->credential"))) {
                    
$result['code'] = AuthenticationResult::SUCCESS;
                } else {
                    
$result['code'] = AuthenticationResult::FAILURE_CREDENTIAL_INVALID;
                    
$result['messages'][] = 'Password incorrect';
                }
                return new 
AuthenticationResult($result['code'], $result['identity'], $result['messages']);
            }
        }

        
$result['code'] = AuthenticationResult::FAILURE_IDENTITY_NOT_FOUND;
        
$result['messages'][] = "Username '$this->identity' and realm '$this->realm' combination not found";
        return new 
AuthenticationResult($result['code'], $result['identity'], $result['messages']);
    }

    
/**
     * Securely compare two strings for equality while avoided C level memcmp()
     * optimisations capable of leaking timing information useful to an attacker
     * attempting to iteratively guess the unknown string (e.g. password) being
     * compared against.
     *
     * @param string $a
     * @param string $b
     * @return bool
     */
    
protected function _secureStringCompare($a$b)
    {
        if (
strlen($a) !== strlen($b)) {
            return 
false;
        }
        
$result 0;
        for (
$i 0$len strlen($a); $i $len$i++) {
            
$result |= ord($a[$i]) ^ ord($b[$i]);
        }
        return 
$result == 0;
    }
}