2010-01-28 05:22:37 +01:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* eGroupWare API - LDAP Authentication with fallback to SQL
|
|
|
|
*
|
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @author Ralf Becker <ralfbecker@outdoor-training.de>
|
|
|
|
* @license http://opensource.org/licenses/lgpl-license.php LGPL - GNU Lesser General Public License
|
|
|
|
* @package api
|
2016-03-06 21:47:10 +01:00
|
|
|
* @subpackage auth
|
2010-01-28 05:22:37 +01:00
|
|
|
* @version $Id$
|
|
|
|
*/
|
|
|
|
|
2016-03-06 21:47:10 +01:00
|
|
|
namespace EGroupware\Api\Auth;
|
|
|
|
|
|
|
|
use EGroupware\Api;
|
|
|
|
|
2010-01-28 05:22:37 +01:00
|
|
|
/**
|
2023-07-27 14:09:17 +02:00
|
|
|
* Authentication against an LDAP Server (or other authentication type) with fallback to SQL
|
2010-01-28 05:22:37 +01:00
|
|
|
*/
|
2016-03-06 21:47:10 +01:00
|
|
|
class Fallback implements Backend
|
2010-01-28 05:22:37 +01:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Primary auth backend
|
2013-05-22 11:02:53 +02:00
|
|
|
*
|
2016-03-06 21:47:10 +01:00
|
|
|
* @var Backend
|
2010-01-28 05:22:37 +01:00
|
|
|
*/
|
|
|
|
private $primary_backend;
|
2013-05-22 11:02:53 +02:00
|
|
|
|
2010-01-28 05:22:37 +01:00
|
|
|
/**
|
|
|
|
* Fallback auth backend
|
2013-05-22 11:02:53 +02:00
|
|
|
*
|
2016-03-06 21:47:10 +01:00
|
|
|
* @var Backend
|
2010-01-28 05:22:37 +01:00
|
|
|
*/
|
|
|
|
private $fallback_backend;
|
2013-05-22 11:02:53 +02:00
|
|
|
|
2010-01-28 05:22:37 +01:00
|
|
|
/**
|
|
|
|
* Constructor
|
2016-03-06 21:47:10 +01:00
|
|
|
*
|
|
|
|
* @param string $primary ='ldap'
|
|
|
|
* @param string $fallback ='sql'
|
2010-01-28 05:22:37 +01:00
|
|
|
*/
|
2016-03-06 21:47:10 +01:00
|
|
|
function __construct($primary='ldap',$fallback='sql')
|
2010-01-28 05:22:37 +01:00
|
|
|
{
|
2023-07-07 16:02:02 +02:00
|
|
|
// do NOT save our backends in session, as we want "fallback" to be saved
|
2023-07-27 14:09:17 +02:00
|
|
|
$this->primary_backend = Api\Auth::backend(str_replace('auth_', '', $primary ?: 'ldap'), false);
|
2013-05-22 11:02:53 +02:00
|
|
|
|
2023-07-27 14:09:17 +02:00
|
|
|
$this->fallback_backend = Api\Auth::backend(str_replace('auth_', '', $fallback ?: 'sql'), false);
|
2010-01-28 05:22:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* authentication against LDAP with fallback to SQL
|
|
|
|
*
|
|
|
|
* @param string $username username of account to authenticate
|
|
|
|
* @param string $passwd corresponding password
|
|
|
|
* @return boolean true if successful authenticated, false otherwise
|
|
|
|
*/
|
|
|
|
function authenticate($username, $passwd, $passwd_type='text')
|
|
|
|
{
|
2023-07-07 16:17:23 +02:00
|
|
|
$ret = $this->primary_backend->authenticate($username, $passwd, $passwd_type);
|
|
|
|
Api\Auth::log(__METHOD__."('$username', ...) primary_backend(".get_class($this->primary_backend).
|
|
|
|
")->authenticate('$username', '".str_repeat('*', strlen($passwd))."', ...) returned ".json_encode($ret));
|
|
|
|
if ($ret)
|
2010-01-28 05:22:37 +01:00
|
|
|
{
|
2016-03-06 21:47:10 +01:00
|
|
|
Api\Cache::setInstance(__CLASS__,'backend_used-'.$username,'primary');
|
2013-05-22 11:02:53 +02:00
|
|
|
// check if fallback has correct password, if not update it
|
|
|
|
if (($account_id = $GLOBALS['egw']->accounts->name2id($username)) &&
|
|
|
|
!$this->fallback_backend->authenticate($username,$passwd, $passwd_type))
|
|
|
|
{
|
|
|
|
$backup_currentapp = $GLOBALS['egw_info']['flags']['currentapp'];
|
|
|
|
$GLOBALS['egw_info']['flags']['currentapp'] = 'admin'; // otherwise
|
2023-07-06 15:50:53 +02:00
|
|
|
$ret = $this->fallback_backend->change_password('', $passwd, $account_id);
|
|
|
|
Api\Auth::log(__METHOD__."('$username', ...) fallback_backend(".get_class($this->fallback_backend).
|
|
|
|
")->change_password('', '".str_repeat('*', strlen($passwd))."', $account_id) returned ".json_encode($ret));
|
2013-05-22 11:02:53 +02:00
|
|
|
$GLOBALS['egw_info']['flags']['currentapp'] = $backup_currentapp;
|
2013-06-26 11:49:30 +02:00
|
|
|
//error_log(__METHOD__."('$username', \$passwd) updated password for #$account_id on fallback ".($ret ? 'successfull' : 'failed!'));
|
2013-05-22 11:02:53 +02:00
|
|
|
}
|
2010-01-28 05:22:37 +01:00
|
|
|
return true;
|
|
|
|
}
|
2023-07-06 15:50:53 +02:00
|
|
|
if (($ret = $this->fallback_backend->authenticate($username,$passwd, $passwd_type)))
|
2010-01-28 05:22:37 +01:00
|
|
|
{
|
2016-03-06 21:47:10 +01:00
|
|
|
Api\Cache::setInstance(__CLASS__,'backend_used-'.$username,'fallback');
|
2010-01-28 05:22:37 +01:00
|
|
|
}
|
2023-07-06 15:50:53 +02:00
|
|
|
Api\Auth::log(__METHOD__."('$username', ...) fallback_backend(".get_class($this->fallback_backend).
|
|
|
|
")->authenticate('$username', '".str_repeat('*', strlen($passwd))."', ...) returned ".json_encode($ret));
|
|
|
|
return $ret;
|
2010-01-28 05:22:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* changes password in LDAP
|
|
|
|
*
|
|
|
|
* If $old_passwd is given, the password change is done binded as user and NOT with the
|
|
|
|
* "root" dn given in the configurations.
|
|
|
|
*
|
|
|
|
* @param string $old_passwd must be cleartext or empty to not to be checked
|
|
|
|
* @param string $new_passwd must be cleartext
|
|
|
|
* @param int $account_id account id of user whose passwd should be changed
|
|
|
|
* @return boolean true if password successful changed, false otherwise
|
|
|
|
*/
|
|
|
|
function change_password($old_passwd, $new_passwd, $account_id=0)
|
|
|
|
{
|
2016-07-27 10:19:15 +02:00
|
|
|
if(!$account_id)
|
2011-09-22 17:29:41 +02:00
|
|
|
{
|
|
|
|
$account_id = $GLOBALS['egw_info']['user']['account_id'];
|
|
|
|
$username = $GLOBALS['egw_info']['user']['account_lid'];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$username = $GLOBALS['egw']->accounts->id2name($account_id);
|
|
|
|
}
|
2016-03-06 21:47:10 +01:00
|
|
|
if (Api\Cache::getInstance(__CLASS__,'backend_used-'.$username) == 'primary')
|
2010-01-28 05:22:37 +01:00
|
|
|
{
|
2014-06-18 10:22:12 +02:00
|
|
|
if (($ret = $this->primary_backend->change_password($old_passwd, $new_passwd, $account_id)))
|
2013-05-22 11:02:53 +02:00
|
|
|
{
|
|
|
|
// if password successfully changed on primary, also update fallback
|
2023-07-06 15:50:53 +02:00
|
|
|
$change_pwd = $this->fallback_backend->change_password($old_passwd, $new_passwd, $account_id);
|
|
|
|
Api\Auth::log(__METHOD__."(..., $account_id) fallback_backend(".get_class($this->fallback_backend).
|
|
|
|
")->change_password('', '".str_repeat('*', strlen($new_passwd))."', $account_id) returned ".json_encode($change_pwd));
|
2013-05-22 11:02:53 +02:00
|
|
|
}
|
2010-01-28 05:22:37 +01:00
|
|
|
}
|
2014-06-18 10:22:12 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
$ret = $this->fallback_backend->change_password($old_passwd, $new_passwd, $account_id);
|
2023-07-06 15:50:53 +02:00
|
|
|
Api\Auth::log(__METHOD__."(..., $account_id) fallback_backend(".get_class($this->fallback_backend).
|
|
|
|
")->change_password('', '".str_repeat('*', strlen($new_passwd))."', $account_id) returned ".json_encode($ret));
|
2014-06-18 10:22:12 +02:00
|
|
|
}
|
2016-03-06 21:47:10 +01:00
|
|
|
//error_log(__METHOD__."('$old_passwd', '$new_passwd', $account_id) username='$username', backend=".Api\Cache::getInstance(__CLASS__,'backend_used-'.$username)." returning ".array2string($ret));
|
2014-06-18 10:22:12 +02:00
|
|
|
return $ret;
|
2010-01-28 05:22:37 +01:00
|
|
|
}
|
2011-09-22 17:29:41 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* fetch the last pwd change for the user
|
|
|
|
*
|
|
|
|
* @param string $username username of account to authenticate
|
|
|
|
* @return mixed false or account_lastpwd_change
|
|
|
|
*/
|
|
|
|
function getLastPwdChange($username)
|
|
|
|
{
|
2016-03-06 21:47:10 +01:00
|
|
|
if (Api\Cache::getInstance(__CLASS__,'backend_used-'.$username) == 'primary')
|
2011-09-22 17:29:41 +02:00
|
|
|
{
|
2016-03-06 21:47:10 +01:00
|
|
|
if (method_exists($this->primary_backend,'getLastPwdChange'))
|
|
|
|
{
|
|
|
|
return $this->primary_backend->getLastPwdChange($username);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (method_exists($this->fallback_backend,'getLastPwdChange'))
|
|
|
|
{
|
|
|
|
return $this->fallback_backend->getLastPwdChange($username);
|
2011-09-22 17:29:41 +02:00
|
|
|
}
|
2011-09-26 13:01:16 +02:00
|
|
|
return false;
|
2011-09-22 17:29:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* changes account_lastpwd_change in sql datababse
|
|
|
|
*
|
|
|
|
* @param int $account_id account id of user whose passwd should be changed
|
|
|
|
* @param string $passwd must be cleartext, usually not used, but may be used to authenticate as user to do the change -> ldap
|
|
|
|
* @param int $lastpwdchange must be a unixtimestamp
|
|
|
|
* @return boolean true if account_lastpwd_change successful changed, false otherwise
|
|
|
|
*/
|
|
|
|
function setLastPwdChange($account_id=0, $passwd=NULL, $lastpwdchange=NULL)
|
|
|
|
{
|
|
|
|
if(!$account_id || $GLOBALS['egw_info']['flags']['currentapp'] == 'login')
|
|
|
|
{
|
|
|
|
$account_id = $GLOBALS['egw_info']['user']['account_id'];
|
|
|
|
$username = $GLOBALS['egw_info']['user']['account_lid'];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$username = $GLOBALS['egw']->accounts->id2name($account_id);
|
|
|
|
}
|
2016-03-06 21:47:10 +01:00
|
|
|
if (Api\Cache::getInstance(__CLASS__,'backend_used-'.$username) == 'primary')
|
|
|
|
{
|
|
|
|
if (method_exists($this->primary_backend,'setLastPwdChange'))
|
|
|
|
{
|
|
|
|
return $this->primary_backend->setLastPwdChange($username);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (method_exists($this->fallback_backend,'setLastPwdChange'))
|
2011-09-22 17:29:41 +02:00
|
|
|
{
|
2016-03-06 21:47:10 +01:00
|
|
|
return $this->fallback_backend->setLastPwdChange($account_id, $passwd, $lastpwdchange);
|
2011-09-22 17:29:41 +02:00
|
|
|
}
|
2011-09-26 13:01:16 +02:00
|
|
|
return false;
|
2011-09-22 17:29:41 +02:00
|
|
|
}
|
2023-07-06 15:50:53 +02:00
|
|
|
}
|