allow to specify multiple ;-separated hosts for database, by default first one is used, on connection failure other ones are tried too

This commit is contained in:
Ralf Becker 2015-07-06 11:15:14 +00:00
parent 7eff188dcf
commit b27dbc617c
2 changed files with 66 additions and 12 deletions

View File

@ -1636,7 +1636,10 @@ class StreamWrapper implements Vfs\StreamWrapperIface
self::$pdo_type = $egw_db->Type; self::$pdo_type = $egw_db->Type;
break; break;
} }
$dsn = self::$pdo_type.':dbname='.$egw_db->Database.($egw_db->Host ? ';host='.$egw_db->Host.($egw_db->Port ? ';port='.$egw_db->Port : '') : ''); // get host used be egw_db
$host = $egw_db->get_host();
$dsn = self::$pdo_type.':dbname='.$egw_db->Database.($host ? ';host='.$host.($egw_db->Port ? ';port='.$egw_db->Port : '') : '');
// check once if pdo extension and DB specific driver is loaded or can be loaded // check once if pdo extension and DB specific driver is loaded or can be loaded
static $pdo_available=null; static $pdo_available=null;
if (is_null($pdo_available)) if (is_null($pdo_available))

View File

@ -306,7 +306,8 @@ class egw_db
* @param string $User name of database user (optional) * @param string $User name of database user (optional)
* @param string $Password password for database user (optional) * @param string $Password password for database user (optional)
* @param string $Type type of database (optional) * @param string $Type type of database (optional)
* @return ADONewConnection * @throws egw_exception_db_connection
* @return ADOConnection
*/ */
function connect($Database = NULL, $Host = NULL, $Port = NULL, $User = NULL, $Password = NULL,$Type = NULL) function connect($Database = NULL, $Host = NULL, $Port = NULL, $User = NULL, $Password = NULL,$Type = NULL)
{ {
@ -339,9 +340,59 @@ class egw_db
{ {
$this->Type = $GLOBALS['egw_info']['server']['db_type']; $this->Type = $GLOBALS['egw_info']['server']['db_type'];
} }
// on connection failure re-try with an other host
// remembering in session which host we used last time
$use_host_from_session = true;
while(($host = $this->get_host(!$use_host_from_session)))
{
try {
//error_log(__METHOD__."() this->Host(s)=$this->Host, n=$n --> host=$host");
return $this->_connect($host);
}
catch(egw_exception_db_connection $e) {
_egw_log_exception($e);
$this->disconnect(); // force a new connect
$this->Type = $this->setupType; // get set to "mysql" for "mysqli"
$use_host_from_session = false; // re-try with next host from list
}
}
throw $e;
}
/**
* Get one of multiple (semicolon-separated) DB-hosts to use
*
* Which host to use is cached in session, default is first one.
*
* @param boolean $next =false true: move to next host
* @return boolean|string hostname or false, if already number-of-hosts plus 2 times called with $next == true
*/
public function get_host($next = false)
{
$hosts = explode(';', $this->Host);
$num_hosts = count($hosts);
$n =& egw_cache::getSession(__CLASS__, $this->Host);
if (!isset($n)) $n = 0;
if ($next && ++$n >= $num_hosts+2)
{
return false;
}
return $hosts[$n % $num_hosts];
}
/**
* Connect to given host
*
* @param string $Host host to connect to
* @return ADOConnection
* @throws egw_exception_db_connection
*/
protected function _connect($Host)
{
if (!$this->Link_ID) if (!$this->Link_ID)
{ {
foreach(array('Host','Database','User','Password') as $name) foreach(array('Database','User','Password') as $name)
{ {
$$name = $this->$name; $$name = $this->$name;
} }