* @link https://github.com/pradosoft/prado
* @copyright Copyright © 2005-2015 The PRADO Group
* @license https://github.com/pradosoft/prado/blob/master/COPYRIGHT
* @package System.Security
*/
/**
* Using TUser class
*/
Prado::using('System.Security.TUser');
/**
* TUserManager class
*
* TUserManager manages a static list of users {@link TUser}.
* The user information is specified via module configuration using the following XML syntax,
*
*
*
*
*
*
*
*
*
* PHP configuration style:
*
* array(
* 'users' => array(
* 'class' => 'System.Security.TUserManager',
* 'properties' => array(
* 'PasswordMode' => 'Clear',
* ),
* 'users' => array(
* array('name'=>'Joe','password'=>'demo'),
* array('name'=>'John','password'=>'demo'),
* ),
* 'roles' => array(
* array('name'=>'Administrator','users'=>'John'),
* array('name'=>'Writer','users'=>'Joe,John'),
* ),
* ),
* )
*
*
* In addition, user information can also be loaded from an external file
* specified by {@link setUserFile UserFile} property. Note, the property
* only accepts a file path in namespace format. The user file format is
* similar to the above sample.
*
* The user passwords may be specified as clear text, SH1 or MD5 hashed by setting
* {@link setPasswordMode PasswordMode} as Clear, SHA1 or MD5.
* The default name for a guest user is Guest. It may be changed
* by setting {@link setGuestName GuestName} property.
*
* TUserManager may be used together with {@link TAuthManager} which manages
* how users are authenticated and authorized in a Prado application.
*
* @author Qiang Xue
* @author Carl Mathisen
* @package System.Security
* @since 3.0
*/
class TUserManager extends TModule implements IUserManager
{
/**
* extension name to the user file
*/
const USER_FILE_EXT='.xml';
/**
* @var array list of users managed by this module
*/
private $_users=array();
/**
* @var array list of roles managed by this module
*/
private $_roles=array();
/**
* @var string guest name
*/
private $_guestName='Guest';
/**
* @var TUserManagerPasswordMode password mode
*/
private $_passwordMode=TUserManagerPasswordMode::MD5;
/**
* @var boolean whether the module has been initialized
*/
private $_initialized=false;
/**
* @var string user/role information file
*/
private $_userFile=null;
/**
* Initializes the module.
* This method is required by IModule and is invoked by application.
* It loads user/role information from the module configuration.
* @param mixed module configuration
*/
public function init($config)
{
$this->loadUserData($config);
if($this->_userFile!==null)
{
if($this->getApplication()->getConfigurationType()==TApplication::CONFIG_TYPE_PHP)
{
$userFile = include $this->_userFile;
$this->loadUserDataFromPhp($userFile);
}
else
{
$dom=new TXmlDocument;
$dom->loadFromFile($this->_userFile);
$this->loadUserDataFromXml($dom);
}
}
$this->_initialized=true;
}
/*
* Loads user/role information
* @param mixed the variable containing the user information
*/
private function loadUserData($config)
{
if($this->getApplication()->getConfigurationType()==TApplication::CONFIG_TYPE_PHP)
$this->loadUserDataFromPhp($config);
else
$this->loadUserDataFromXml($config);
}
/**
* Loads user/role information from an php array.
* @param array the array containing the user information
*/
private function loadUserDataFromPhp($config)
{
if(isset($config['users']) && is_array($config['users']))
{
foreach($config['users'] as $user)
{
$name = trim(strtolower(isset($user['name'])?$user['name']:''));
$password = isset($user['password'])?$user['password']:'';
$this->_users[$name] = $password;
$roles = isset($user['roles'])?$user['roles']:'';
if($roles!=='')
{
foreach(explode(',',$roles) as $role)
{
if(($role=trim($role))!=='')
$this->_roles[$name][]=$role;
}
}
}
}
if(isset($config['roles']) && is_array($config['roles']))
{
foreach($config['roles'] as $role)
{
$name = isset($role['name'])?$role['name']:'';
$users = isset($role['users'])?$role['users']:'';
foreach(explode(',',$users) as $user)
{
if(($user=trim($user))!=='')
$this->_roles[strtolower($user)][]=$name;
}
}
}
}
/**
* Loads user/role information from an XML node.
* @param TXmlElement the XML node containing the user information
*/
private function loadUserDataFromXml($xmlNode)
{
foreach($xmlNode->getElementsByTagName('user') as $node)
{
$name=trim(strtolower($node->getAttribute('name')));
$this->_users[$name]=$node->getAttribute('password');
if(($roles=trim($node->getAttribute('roles')))!=='')
{
foreach(explode(',',$roles) as $role)
{
if(($role=trim($role))!=='')
$this->_roles[$name][]=$role;
}
}
}
foreach($xmlNode->getElementsByTagName('role') as $node)
{
foreach(explode(',',$node->getAttribute('users')) as $user)
{
if(($user=trim($user))!=='')
$this->_roles[strtolower($user)][]=$node->getAttribute('name');
}
}
}
/**
* Returns an array of all users.
* Each array element represents a single user.
* The array key is the username in lower case, and the array value is the
* corresponding user password.
* @return array list of users
*/
public function getUsers()
{
return $this->_users;
}
/**
* Returns an array of user role information.
* Each array element represents the roles for a single user.
* The array key is the username in lower case, and the array value is
* the roles (represented as an array) that the user is in.
* @return array list of user role information
*/
public function getRoles()
{
return $this->_roles;
}
/**
* @return string the full path to the file storing user/role information
*/
public function getUserFile()
{
return $this->_userFile;
}
/**
* @param string user/role data file path (in namespace form). The file format is XML
* whose content is similar to that user/role block in application configuration.
* @throws TInvalidOperationException if the module is already initialized
* @throws TConfigurationException if the file is not in proper namespace format
*/
public function setUserFile($value)
{
if($this->_initialized)
throw new TInvalidOperationException('usermanager_userfile_unchangeable');
else if(($this->_userFile=Prado::getPathOfNamespace($value,self::USER_FILE_EXT))===null || !is_file($this->_userFile))
throw new TConfigurationException('usermanager_userfile_invalid',$value);
}
/**
* @return string guest name, defaults to 'Guest'
*/
public function getGuestName()
{
return $this->_guestName;
}
/**
* @param string name to be used for guest users.
*/
public function setGuestName($value)
{
$this->_guestName=$value;
}
/**
* @return TUserManagerPasswordMode how password is stored, clear text, or MD5 or SHA1 hashed. Default to TUserManagerPasswordMode::MD5.
*/
public function getPasswordMode()
{
return $this->_passwordMode;
}
/**
* @param TUserManagerPasswordMode how password is stored, clear text, or MD5 or SHA1 hashed.
*/
public function setPasswordMode($value)
{
$this->_passwordMode=TPropertyValue::ensureEnum($value,'TUserManagerPasswordMode');
}
/**
* Validates if the username and password are correct.
* @param string user name
* @param string password
* @return boolean true if validation is successful, false otherwise.
*/
public function validateUser($username,$password)
{
if($this->_passwordMode===TUserManagerPasswordMode::MD5)
$password=md5($password);
else if($this->_passwordMode===TUserManagerPasswordMode::SHA1)
$password=sha1($password);
$username=strtolower($username);
return (isset($this->_users[$username]) && $this->_users[$username]===$password);
}
/**
* Returns a user instance given the user name.
* @param string user name, null if it is a guest.
* @return TUser the user instance, null if the specified username is not in the user database.
*/
public function getUser($username=null)
{
if($username===null)
{
$user=new TUser($this);
$user->setIsGuest(true);
return $user;
}
else
{
$username=strtolower($username);
if(isset($this->_users[$username]))
{
$user=new TUser($this);
$user->setName($username);
$user->setIsGuest(false);
if(isset($this->_roles[$username]))
$user->setRoles($this->_roles[$username]);
return $user;
}
else
return null;
}
}
/**
* Returns a user instance according to auth data stored in a cookie.
* @param THttpCookie the cookie storing user authentication information
* @return TUser the user instance generated based on the cookie auth data, null if the cookie does not have valid auth data.
* @since 3.1.1
*/
public function getUserFromCookie($cookie)
{
if(($data=$cookie->getValue())!=='')
{
$data=unserialize($data);
if(is_array($data) && count($data)===2)
{
list($username,$token)=$data;
if(isset($this->_users[$username]) && $token===md5($username.$this->_users[$username]))
return $this->getUser($username);
}
}
return null;
}
/**
* Saves user auth data into a cookie.
* @param THttpCookie the cookie to receive the user auth data.
* @since 3.1.1
*/
public function saveUserToCookie($cookie)
{
$user=$this->getApplication()->getUser();
$username=strtolower($user->getName());
if(isset($this->_users[$username]))
{
$data=array($username,md5($username.$this->_users[$username]));
$cookie->setValue(serialize($data));
}
}
/**
* Sets a user as a guest.
* User name is changed as guest name, and roles are emptied.
* @param TUser the user to be changed to a guest.
*/
public function switchToGuest($user)
{
$user->setIsGuest(true);
}
}
/**
* TUserManagerPasswordMode class.
* TUserManagerPasswordMode defines the enumerable type for the possible modes
* that user passwords can be specified for a {@link TUserManager}.
*
* The following enumerable values are defined:
* - Clear: the password is in plain text
* - MD5: the password is recorded as the MD5 hash value of the original password
* - SHA1: the password is recorded as the SHA1 hash value of the original password
*
* @author Qiang Xue
* @package System.Security
* @since 3.0.4
*/
class TUserManagerPasswordMode extends TEnumerable
{
const Clear='Clear';
const MD5='MD5';
const SHA1='SHA1';
}