enygma/yubikey

View on GitHub
src/Yubikey/Response.php

Summary

Maintainability
A
0 mins
Test Coverage
<?php

namespace Yubikey;

class Response
{
    /**
     * Returned value of the hash given in the request
     * @var string
     */
    private $h = null;

    /**
     * Timestamp in UTC of request
     * @var string
     */
    private $t = null;

    /**
     * OTP given in request
     * @var string
     */
    private $otp = null;

    /**
     * Nonce given in the request
     * @var string
     */
    private $nonce = null;

    /**
     * Percent of validation servers that replied with "success" (0% - 100%)
     * @var integer
     */
    private $sl = null;

    /**
     * Status of the response (see constants below)
     * @var string
     */
    private $status = null;

    /**
     * OTP given by the user (used for verification)
     * @var string
     */
    private $inputOtp = null;

    /**
     * Nonce used in the request (used for verification)
     * @var string
     */
    private $inputNonce = null;

    /**
     * Timestamp returned from the response
     * @var string
     */
    private $timestamp = null;

    /**
     * Session counter
     * @var integer
     */
    private $sessioncounter = null;

    /**
     * Session use #
     * @var integer
     */
    private $sessionuse = null;

    /**
     * Hostname request was made to
     * @var string
     */
    private $host;

    /**
     * Microtime difference taken to get response
     * @var integer
     */
    private $mt;

    /**
     * Define constants for the return status from API
     */
    const SUCCESS = 'OK';
    const REPLAY_OTP = 'REPLAYED_OTP';
    const REPLAY_REQUEST = 'REPLAYED_REQUEST';
    const MISSING_PARAMETER = 'MISSING_PARAMETER';
    const NO_CLIENT = 'NO_SUCH_CLIENT';
    const BAD_OTP = 'BAD_OTP';
    const BAD_SIGNATURE = 'BAD_SIGNATURE';
    const OP_NOT_ALLOWED = 'OPERATION_NOT_ALLOWED';
    const BACKEND_ERROR = 'BACKEND_ERROR';
    const NOT_ENOUGH_ANSWERS = 'NOT_ENOUGH_ANSWERS';

    /**
     * Init the object and set the data into the response
     *
     * @param array $data Data from the Yubi API response
     */
    public function __construct($data = null)
    {
        if ($data !== null) {
            $this->load($data);
        }
    }

    /**
     * Load the data into the object
     *
     * @param array $data Data from the object
     * @return boolean True when loading is done
     */
    public function load($data)
    {
        foreach ($data as $index => $value) {
            if (property_exists($this, $index)) {
                $this->$index = trim($value);
            }
        }
        return true;
    }

    /**
     * Parse the return data from the request and
     *     load it into the object properties
     *
     * @param string $data API return data
     */
    public function parse($data)
    {
        $result = array();
        $parts = explode("\n", $data);

        foreach($parts as $index => $part) {
            $kv = explode("=", $part);
            if (!empty($kv[1])) {
                $result[$kv[0]] = $kv[1];
            }
        }

        $this->load($result);
    }

    /**
     * Get the time value for the response
     *
     * @return string Date/time string
     */
    public function getTime()
    {
        return $this->t;
    }

    /**
     * Return the time to execute (microtime)
     *
     * @return integer Time result
     */
    public function getMt()
    {
        return $this->mt;
    }

    /**
     * Set the OTP used in the request
     * @param string $otp OTP string (from key)
     */
    public function setInputOtp($otp)
    {
        $this->inputOtp = $otp;
        return $this;
    }

    /**
     * Get the OTP used in the request
     * @return string OTP string
     */
    public function getInputOtp()
    {
        return $this->inputOtp;
    }

    /**
     * Set the nonce used in the request
     * @param string $nonce Nonce from request
     */
    public function setInputNonce($nonce)
    {
        $this->inputNonce = $nonce;
        return $this;
    }

    /**
     * Get the nonce used in the request
     * @return string Nonce from request
     */
    public function getInputNonce()
    {
        return $this->inputNonce;
    }

    /**
     * Set response hostname
     * @param string $host Hostname
     */
    public function setHost($host)
    {
        $this->host = $host;
        return $this;
    }

    /**
     * Get the current hostname
     * @return string
     */
    public function getHost()
    {
        return $this->host;
    }

    /**
     * Get the hash from the response
     *
     * @param boolean $encode "Encode" the data (replace + with %2B)
     * @return string Hash value
     */
    public function getHash($encode = false)
    {
        $hash = $this->h;
        if (substr($hash, -1) !== '=') {
            $hash .= '=';
        }
        if ($encode === true) {
            $hash = str_replace('+', '%2B', $hash);
        }
        return $hash;
    }

    /**
     * Get the properties of the response
     *
     * @return array Response property list
     */
    public function getProperties()
    {
        return array(
            't', 'otp', 'nonce', 'sl', 'status',
            'timestamp', 'sessioncounter', 'sessionuse'
        );
    }

    /**
     * Magic method to get access to the private properties
     * @param string $name Property name
     * @return string|null If found, returns teh value. If not, null
     */
    public function __get($name)
    {
        return (property_exists($this, $name)) ? $this->$name : null;
    }

    /**
     * Check the success of the response
     *     Validates: status, OTP and nonce
     * @return boolean Success/fail of request
     */
    public function success()
    {
        $inputOtp = $this->getInputOtp();
        $inputNonce = $this->getInputNonce();

        if ($inputNonce === null || $inputOtp === null) {
            return false;
        }

        return (
            $inputOtp == $this->otp
            && $inputNonce === $this->nonce
            && $this->status === Response::SUCCESS
        ) ? true : false;
    }
}