You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
117 lines
4.1 KiB
117 lines
4.1 KiB
10 months ago
|
<?php
|
||
|
class LtDbConnectionManager
|
||
|
{
|
||
|
/**
|
||
|
* Connection management
|
||
|
* array(
|
||
|
* "connection" => connection resource id,
|
||
|
* "expire_time" => expire time,
|
||
|
* "schema" => default schema name,
|
||
|
* "charset" => char set / encoding
|
||
|
* )
|
||
|
*/
|
||
|
static public $connectionPool;
|
||
|
public $configHandle;
|
||
|
protected $connectionAdapter;
|
||
|
protected $sqlAdapter;
|
||
|
private $servers;
|
||
|
|
||
|
public function getConnection($group, $node, $role = "master")
|
||
|
{
|
||
|
if(empty($this->servers))
|
||
|
{
|
||
|
$this->servers = $this->configHandle->get("db.servers");
|
||
|
}
|
||
|
if (($connection = $this->getNewConnection($group, $node, $role)) ||($connection = $this->getCachedConnection($group, $node, $role)))
|
||
|
{
|
||
|
return array(
|
||
|
"connectionAdapter" => $this->connectionAdapter,
|
||
|
"connectionResource" => $connection
|
||
|
);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
trigger_error("db server can not be connected: group=$group, node=$node, role=$role", E_USER_ERROR);
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected function getConnectionKey($connConf)
|
||
|
{
|
||
|
return $connConf['adapter'] . $connConf['host'] . $connConf['port'] . $connConf['username'] . $connConf['dbname'];
|
||
|
}
|
||
|
|
||
|
protected function saveConnection($connConf, $connection, $ttl)
|
||
|
{
|
||
|
$connectionInfo = array(
|
||
|
"connection" => $connection,
|
||
|
"expire_time" => time() + $ttl,
|
||
|
"schema" => $connConf["schema"],
|
||
|
"charset" => $connConf["charset"],
|
||
|
);
|
||
|
self::$connectionPool[$this->getConnectionKey($connConf)] = $connectionInfo;
|
||
|
}
|
||
|
|
||
|
protected function getCachedConnection($group, $node, $role)
|
||
|
{
|
||
|
foreach($this->servers[$group][$node][$role] as $hostConfig)
|
||
|
{
|
||
|
$key = $this->getConnectionKey($hostConfig);
|
||
|
if(isset(self::$connectionPool[$key]) && time() < self::$connectionPool[$key]['expire_time'])
|
||
|
{//cached connection resource FOUND
|
||
|
$connectionInfo = self::$connectionPool[$key];
|
||
|
if ($connectionInfo["schema"] != $hostConfig["schema"] || $connectionInfo["charset"] != $hostConfig["charset"])
|
||
|
{//检查当前schema和charset与用户要操作的目标不一致
|
||
|
$hostConfig = $this->servers[$group][$node][$role][$hostIndexArray[$hashNumber]];
|
||
|
$dbFactory = new LtDbAdapterFactory;
|
||
|
$this->connectionAdapter = $dbFactory->getConnectionAdapter($hostConfig["connection_adapter"]);
|
||
|
$this->sqlAdapter = $dbFactory->getSqlAdapter($hostConfig["sql_adapter"]);
|
||
|
if ($connectionInfo["schema"] != $hostConfig["schema"])
|
||
|
{
|
||
|
$this->connectionAdapter->exec($this->sqlAdapter->setSchema($hostConfig["schema"]), $connectionInfo["connection"]);
|
||
|
}
|
||
|
if ($connectionInfo["charset"] != $hostConfig["charset"])
|
||
|
{
|
||
|
$this->connectionAdapter->exec($this->sqlAdapter->setCharset($hostConfig["charset"]), $connectionInfo["connection"]);
|
||
|
}
|
||
|
$this->saveConnection($hostConfig, $connectionInfo["connection"], $hostConfig["connection_ttl"]);
|
||
|
}
|
||
|
return $connectionInfo["connection"];
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
protected function getNewConnection($group, $node, $role)
|
||
|
{
|
||
|
$hostTotal = count($this->servers[$group][$node][$role]);
|
||
|
$hostIndexArray = array_keys($this->servers[$group][$node][$role]);
|
||
|
while ($hostTotal)
|
||
|
{
|
||
|
$hashNumber = substr(microtime(),7,1) % $hostTotal;
|
||
|
$hostConfig = $this->servers[$group][$node][$role][$hostIndexArray[$hashNumber]];
|
||
|
$dbFactory = new LtDbAdapterFactory;
|
||
|
$this->connectionAdapter = $dbFactory->getConnectionAdapter($hostConfig["connection_adapter"]);
|
||
|
$this->sqlAdapter = $dbFactory->getSqlAdapter($hostConfig["sql_adapter"]);
|
||
|
if ($connection = $this->connectionAdapter->connect($hostConfig))
|
||
|
{
|
||
|
$this->connectionAdapter->exec($this->sqlAdapter->setSchema($hostConfig["schema"]), $connection);
|
||
|
$this->connectionAdapter->exec($this->sqlAdapter->setCharset($hostConfig["charset"]), $connection);
|
||
|
$this->saveConnection($hostConfig, $connection, $hostConfig["connection_ttl"]);
|
||
|
return $connection;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//trigger_error('connection fail', E_USER_WARNING);
|
||
|
//delete the unavailable server
|
||
|
for ($i = $hashNumber; $i < $hostTotal - 1; $i ++)
|
||
|
{
|
||
|
$hostIndexArray[$i] = $hostIndexArray[$i+1];
|
||
|
}
|
||
|
unset($hostIndexArray[$hostTotal-1]);
|
||
|
$hostTotal --;
|
||
|
}//end else
|
||
|
}//end while
|
||
|
return false;
|
||
|
}
|
||
|
}
|