209 lines
6.0 KiB
PHP
209 lines
6.0 KiB
PHP
|
<?php
|
||
|
|
||
|
/**
|
||
|
* Supplies Redis server store backend for OpenID servers and consumers.
|
||
|
* Uses Predis library {@see https://github.com/nrk/predis}.
|
||
|
* Requires PHP >= 5.3.
|
||
|
*
|
||
|
* LICENSE: See the COPYING file included in this distribution.
|
||
|
*
|
||
|
* @package OpenID
|
||
|
* @author Ville Mattila <ville@eventio.fi>
|
||
|
* @copyright 2008 JanRain Inc., 2013 Eventio Oy / Ville Mattila
|
||
|
* @license http://www.apache.org/licenses/LICENSE-2.0 Apache
|
||
|
* Contributed by Eventio Oy <http://www.eventio.fi/>
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Import the interface for creating a new store class.
|
||
|
*/
|
||
|
require_once 'Auth/OpenID/Interface.php';
|
||
|
|
||
|
/**
|
||
|
* Supplies Redis server store backend for OpenID servers and consumers.
|
||
|
* Uses Predis library {@see https://github.com/nrk/predis}.
|
||
|
* Requires PHP >= 5.3.
|
||
|
*
|
||
|
* @package OpenID
|
||
|
*/
|
||
|
class Auth_OpenID_PredisStore extends Auth_OpenID_OpenIDStore {
|
||
|
|
||
|
/**
|
||
|
* @var \Predis\Client
|
||
|
*/
|
||
|
protected $redis;
|
||
|
|
||
|
/**
|
||
|
* Prefix for Redis keys
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $prefix;
|
||
|
|
||
|
/**
|
||
|
* Initializes a new {@link Auth_OpenID_PredisStore} instance.
|
||
|
*
|
||
|
* @param \Predis\Client $redis Predis client object
|
||
|
* @param string $prefix Prefix for all keys stored to the Redis
|
||
|
*/
|
||
|
function Auth_OpenID_PredisStore(\Predis\Client $redis, $prefix = '')
|
||
|
{
|
||
|
$this->prefix = $prefix;
|
||
|
$this->redis = $redis;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Store association until its expiration time in Redis server.
|
||
|
* Overwrites any existing association with same server_url and
|
||
|
* handle. Handles list of associations for every server.
|
||
|
*/
|
||
|
function storeAssociation($server_url, $association)
|
||
|
{
|
||
|
// create Redis keys for association itself
|
||
|
// and list of associations for this server
|
||
|
$associationKey = $this->associationKey($server_url,
|
||
|
$association->handle);
|
||
|
$serverKey = $this->associationServerKey($server_url);
|
||
|
|
||
|
// save association to server's associations' keys list
|
||
|
$this->redis->lpush(
|
||
|
$serverKey,
|
||
|
$associationKey
|
||
|
);
|
||
|
|
||
|
// Will touch the association list expiration, to avoid filling up
|
||
|
$newExpiration = ($association->issued + $association->lifetime);
|
||
|
|
||
|
$expirationKey = $serverKey.'_expires_at';
|
||
|
$expiration = $this->redis->get($expirationKey);
|
||
|
if (!$expiration || $newExpiration > $expiration) {
|
||
|
$this->redis->set($expirationKey, $newExpiration);
|
||
|
$this->redis->expireat($serverKey, $newExpiration);
|
||
|
$this->redis->expireat($expirationKey, $newExpiration);
|
||
|
}
|
||
|
|
||
|
// save association itself, will automatically expire
|
||
|
$this->redis->setex(
|
||
|
$associationKey,
|
||
|
$newExpiration - time(),
|
||
|
serialize($association)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Read association from Redis. If no handle given
|
||
|
* and multiple associations found, returns latest issued
|
||
|
*/
|
||
|
function getAssociation($server_url, $handle = null)
|
||
|
{
|
||
|
// simple case: handle given
|
||
|
if ($handle !== null) {
|
||
|
return $this->getAssociationFromServer(
|
||
|
$this->associationKey($server_url, $handle)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// no handle given, receiving the latest issued
|
||
|
$serverKey = $this->associationServerKey($server_url);
|
||
|
$lastKey = $this->redis->lpop($serverKey);
|
||
|
if (!$lastKey) { return null; }
|
||
|
|
||
|
// get association, return null if failed
|
||
|
return $this->getAssociationFromServer($lastKey);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Function to actually receive and unserialize the association
|
||
|
* from the server.
|
||
|
*/
|
||
|
private function getAssociationFromServer($associationKey)
|
||
|
{
|
||
|
$association = $this->redis->get($associationKey);
|
||
|
return $association ? unserialize($association) : null;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Immediately delete association from Redis.
|
||
|
*/
|
||
|
function removeAssociation($server_url, $handle)
|
||
|
{
|
||
|
// create Redis keys
|
||
|
$serverKey = $this->associationServerKey($server_url);
|
||
|
$associationKey = $this->associationKey($server_url,
|
||
|
$handle);
|
||
|
|
||
|
// Removing the association from the server's association list
|
||
|
$removed = $this->redis->lrem($serverKey, 0, $associationKey);
|
||
|
if ($removed < 1) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Delete the association itself
|
||
|
return $this->redis->del($associationKey);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create nonce for server and salt, expiring after
|
||
|
* $Auth_OpenID_SKEW seconds.
|
||
|
*/
|
||
|
function useNonce($server_url, $timestamp, $salt)
|
||
|
{
|
||
|
global $Auth_OpenID_SKEW;
|
||
|
|
||
|
// save one request to memcache when nonce obviously expired
|
||
|
if (abs($timestamp - time()) > $Auth_OpenID_SKEW) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// SETNX will set the value only of the key doesn't exist yet.
|
||
|
$nonceKey = $this->nonceKey($server_url, $salt);
|
||
|
$added = $this->predis->setnx($nonceKey);
|
||
|
if ($added) {
|
||
|
// Will set expiration
|
||
|
$this->predis->expire($nonceKey, $Auth_OpenID_SKEW);
|
||
|
return true;
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Build up nonce key
|
||
|
*/
|
||
|
private function nonceKey($server_url, $salt)
|
||
|
{
|
||
|
return $this->prefix .
|
||
|
'openid_nonce_' .
|
||
|
sha1($server_url) . '_' . sha1($salt);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Key is prefixed with $prefix and 'openid_association_' string
|
||
|
*/
|
||
|
function associationKey($server_url, $handle = null)
|
||
|
{
|
||
|
return $this->prefix .
|
||
|
'openid_association_' .
|
||
|
sha1($server_url) . '_' . sha1($handle);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Key is prefixed with $prefix and 'openid_association_server_' string
|
||
|
*/
|
||
|
function associationServerKey($server_url)
|
||
|
{
|
||
|
return $this->prefix .
|
||
|
'openid_association_server_' .
|
||
|
sha1($server_url);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Report that this storage doesn't support cleanup
|
||
|
*/
|
||
|
function supportsCleanup()
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|