mirror of
https://github.com/danog/sql.git
synced 2024-11-30 04:29:18 +01:00
Proper AbstractPool and add PooledStatement
This commit is contained in:
parent
fe530cf590
commit
15336ca7fc
@ -2,72 +2,70 @@
|
|||||||
|
|
||||||
namespace Amp\Sql;
|
namespace Amp\Sql;
|
||||||
|
|
||||||
use function Amp\call;
|
|
||||||
use Amp\CallableMaker;
|
use Amp\CallableMaker;
|
||||||
use function Amp\coroutine;
|
|
||||||
use Amp\Deferred;
|
use Amp\Deferred;
|
||||||
use Amp\Loop;
|
use Amp\Loop;
|
||||||
use Amp\Promise;
|
use Amp\Promise;
|
||||||
|
use function Amp\call;
|
||||||
|
use function Amp\coroutine;
|
||||||
|
|
||||||
abstract class AbstractPool implements Pool
|
abstract class AbstractPool implements Pool
|
||||||
{
|
{
|
||||||
use CallableMaker;
|
use CallableMaker;
|
||||||
|
|
||||||
/** @var Connector */
|
/** @var Connector */
|
||||||
protected $connector;
|
private $connector;
|
||||||
|
|
||||||
/** @var ConnectionConfig */
|
/** @var ConnectionConfig */
|
||||||
protected $config;
|
private $connectionConfig;
|
||||||
|
|
||||||
/** @var int */
|
/** @var int */
|
||||||
protected $maxConnections;
|
private $maxConnections;
|
||||||
|
|
||||||
/** @var \SplQueue */
|
/** @var \SplQueue */
|
||||||
protected $idle;
|
private $idle;
|
||||||
|
|
||||||
/** @var \SplObjectStorage */
|
/** @var \SplObjectStorage */
|
||||||
protected $connections;
|
private $connections;
|
||||||
|
|
||||||
/** @var bool */
|
/** @var \Amp\Promise|null */
|
||||||
protected $closed = false;
|
private $promise;
|
||||||
|
|
||||||
/** @var string */
|
/** @var \Amp\Deferred|null */
|
||||||
protected $timeoutWatcher;
|
private $deferred;
|
||||||
|
|
||||||
/** @var int */
|
|
||||||
protected $idleTimeout = Pool::DEFAULT_IDLE_TIMEOUT;
|
|
||||||
|
|
||||||
/** @var callable */
|
/** @var callable */
|
||||||
protected $prepare;
|
private $prepare;
|
||||||
|
|
||||||
/** @var int */
|
/** @var int */
|
||||||
protected $lastUsedAt;
|
private $pending = 0;
|
||||||
|
|
||||||
/** @var Promise|null */
|
/** @var int */
|
||||||
protected $promise;
|
private $idleTimeout = self::DEFAULT_IDLE_TIMEOUT;
|
||||||
|
|
||||||
/** @var Deferred|null */
|
/** @var string */
|
||||||
protected $deferred;
|
private $timeoutWatcher;
|
||||||
|
|
||||||
/** @var int Number of pending connections. */
|
/** @var bool */
|
||||||
protected $pending = 0;
|
private $closed = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param ConnectionConfig $config
|
* Create a default connector object based on the library of the extending class.
|
||||||
* @param int $maxConnections
|
|
||||||
* @param Connector $connector
|
|
||||||
*
|
*
|
||||||
* @throws \Error If $maxConnections is less than 1.
|
* @return Connector
|
||||||
*/
|
*/
|
||||||
|
abstract protected function createDefaultConnector(): Connector;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
ConnectionConfig $config,
|
ConnectionConfig $config,
|
||||||
int $maxConnections = Pool::DEFAULT_MAX_CONNECTIONS,
|
int $maxConnections = self::DEFAULT_MAX_CONNECTIONS,
|
||||||
Connector $connector = null
|
Connector $connector = null
|
||||||
) {
|
) {
|
||||||
$this->connector = $connector ?? $this->defaultConnector();
|
$this->connector = $connector ?? $this->createDefaultConnector();
|
||||||
$this->config = $config;
|
|
||||||
$this->maxConnections = $maxConnections;
|
|
||||||
|
|
||||||
|
$this->connectionConfig = $config;
|
||||||
|
|
||||||
|
$this->maxConnections = $maxConnections;
|
||||||
if ($this->maxConnections < 1) {
|
if ($this->maxConnections < 1) {
|
||||||
throw new \Error("Pool must contain at least one connection");
|
throw new \Error("Pool must contain at least one connection");
|
||||||
}
|
}
|
||||||
@ -81,8 +79,8 @@ abstract class AbstractPool implements Pool
|
|||||||
$this->timeoutWatcher = Loop::repeat(1000, static function () use (&$idleTimeout, $connections, $idle) {
|
$this->timeoutWatcher = Loop::repeat(1000, static function () use (&$idleTimeout, $connections, $idle) {
|
||||||
$now = \time();
|
$now = \time();
|
||||||
while (!$idle->isEmpty()) {
|
while (!$idle->isEmpty()) {
|
||||||
/** @var Connection $connection */
|
|
||||||
$connection = $idle->bottom();
|
$connection = $idle->bottom();
|
||||||
|
\assert($connection instanceof Link);
|
||||||
|
|
||||||
if ($connection->lastUsedAt() + $idleTimeout > $now) {
|
if ($connection->lastUsedAt() + $idleTimeout > $now) {
|
||||||
return;
|
return;
|
||||||
@ -96,8 +94,6 @@ abstract class AbstractPool implements Pool
|
|||||||
});
|
});
|
||||||
|
|
||||||
Loop::unreference($this->timeoutWatcher);
|
Loop::unreference($this->timeoutWatcher);
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __destruct()
|
public function __destruct()
|
||||||
@ -105,143 +101,6 @@ abstract class AbstractPool implements Pool
|
|||||||
Loop::cancel($this->timeoutWatcher);
|
Loop::cancel($this->timeoutWatcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function query(string $sql): Promise
|
|
||||||
{
|
|
||||||
return call(function () use ($sql) {
|
|
||||||
/** @var Connection $connection */
|
|
||||||
$connection = yield from $this->pop();
|
|
||||||
|
|
||||||
try {
|
|
||||||
$result = yield $connection->query($sql);
|
|
||||||
} catch (\Throwable $exception) {
|
|
||||||
$this->push($connection);
|
|
||||||
throw $exception;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($result instanceof Operation) {
|
|
||||||
$result->onDestruct(function () use ($connection) {
|
|
||||||
$this->push($connection);
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
$this->push($connection);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public function prepare(string $sql): Promise
|
|
||||||
{
|
|
||||||
return call(function () use ($sql) {
|
|
||||||
$statement = yield from $this->doPrepare($sql);
|
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
|
|
||||||
return $this->newPooledStatement($this, $statement, $this->prepare);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public function execute(string $sql, array $params = []): Promise
|
|
||||||
{
|
|
||||||
return call(function () use ($sql, $params) {
|
|
||||||
/** @var Connection $connection */
|
|
||||||
$connection = yield from $this->pop();
|
|
||||||
|
|
||||||
try {
|
|
||||||
$result = yield $connection->execute($sql, $params);
|
|
||||||
} catch (\Throwable $exception) {
|
|
||||||
$this->push($connection);
|
|
||||||
throw $exception;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($result instanceof Operation) {
|
|
||||||
$result->onDestruct(function () use ($connection) {
|
|
||||||
$this->push($connection);
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
$this->push($connection);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public function isAlive(): bool
|
|
||||||
{
|
|
||||||
return !$this->closed;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function close()
|
|
||||||
{
|
|
||||||
// TODO: Implement close() method.
|
|
||||||
}
|
|
||||||
|
|
||||||
public function lastUsedAt(): int
|
|
||||||
{
|
|
||||||
return $this->lastUsedAt;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function transaction(int $isolation = Transaction::ISOLATION_COMMITTED): Promise
|
|
||||||
{
|
|
||||||
return call(function () use ($isolation) {
|
|
||||||
/** @var Connection $connection */
|
|
||||||
$connection = yield from $this->pop();
|
|
||||||
|
|
||||||
try {
|
|
||||||
/** @var Transaction $transaction */
|
|
||||||
$transaction = yield $connection->transaction($isolation);
|
|
||||||
} catch (\Throwable $exception) {
|
|
||||||
$this->push($connection);
|
|
||||||
throw $exception;
|
|
||||||
}
|
|
||||||
|
|
||||||
$transaction->onDestruct(function () use ($connection) {
|
|
||||||
$this->push($connection);
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
|
|
||||||
return $transaction;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Extracts an idle connection from the pool. The connection is completely removed from the pool and cannot be
|
|
||||||
* put back into the pool. Useful for operations where connection state must be changed.
|
|
||||||
*
|
|
||||||
* @return Promise<Connection>
|
|
||||||
*/
|
|
||||||
public function extractConnection(): Promise
|
|
||||||
{
|
|
||||||
return call(function () {
|
|
||||||
$connection = yield from $this->pop();
|
|
||||||
$this->connections->detach($connection);
|
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
|
|
||||||
return $connection;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getConnectionCount(): int
|
|
||||||
{
|
|
||||||
return $this->connections->count();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIdleConnectionCount(): int
|
|
||||||
{
|
|
||||||
return $this->idle->count();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getMaxConnections(): int
|
|
||||||
{
|
|
||||||
return $this->maxConnections;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIdleTimeout(): int
|
public function getIdleTimeout(): int
|
||||||
{
|
{
|
||||||
return $this->idleTimeout;
|
return $this->idleTimeout;
|
||||||
@ -256,38 +115,275 @@ abstract class AbstractPool implements Pool
|
|||||||
$this->idleTimeout = $timeout;
|
$this->idleTimeout = $timeout;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function doPrepare(string $sql): \Generator
|
public function lastUsedAt(): int
|
||||||
{
|
{
|
||||||
/** @var Connection $connection */
|
// Simple implementation... can be improved if needed.
|
||||||
$connection = yield from $this->pop();
|
|
||||||
|
|
||||||
try {
|
$time = 0;
|
||||||
/** @var Statement $statement */
|
|
||||||
$statement = yield $connection->prepare($sql);
|
foreach ($this->connections as $connection) {
|
||||||
} catch (\Throwable $exception) {
|
\assert($connection instanceof Link);
|
||||||
$this->push($connection);
|
if (($lastUsedAt = $connection->lastUsedAt()) > $time) {
|
||||||
throw $exception;
|
$time = $lastUsedAt;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
\assert(
|
return $time;
|
||||||
$statement instanceof Operation,
|
|
||||||
Statement::class . " instances returned from connections must implement " . Operation::class
|
|
||||||
);
|
|
||||||
|
|
||||||
$statement->onDestruct(function () use ($connection) {
|
|
||||||
$this->push($connection);
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->lastUsedAt = \time();
|
|
||||||
|
|
||||||
return $statement;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract protected function pop(): \Generator;
|
/**
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function isAlive(): bool
|
||||||
|
{
|
||||||
|
return !$this->closed;
|
||||||
|
}
|
||||||
|
|
||||||
abstract protected function push(Connection $connection);
|
/**
|
||||||
|
* Close all connections in the pool. No further queries may be made after a pool is closed.
|
||||||
|
*/
|
||||||
|
public function close()
|
||||||
|
{
|
||||||
|
$this->closed = true;
|
||||||
|
foreach ($this->connections as $connection) {
|
||||||
|
$connection->close();
|
||||||
|
}
|
||||||
|
$this->idle = new \SplQueue;
|
||||||
|
$this->connections = new \SplObjectStorage;
|
||||||
|
$this->prepare = null;
|
||||||
|
}
|
||||||
|
|
||||||
abstract protected function defaultConnector(): Connector;
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function extractConnection(): Promise
|
||||||
|
{
|
||||||
|
return call(function () {
|
||||||
|
$connection = yield from $this->pop();
|
||||||
|
$this->connections->detach($connection);
|
||||||
|
return $connection;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
abstract protected function newPooledStatement(Pool $pool, Statement $statement, callable $prepare): Statement;
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function getConnectionCount(): int
|
||||||
|
{
|
||||||
|
return $this->connections->count();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function getIdleConnectionCount(): int
|
||||||
|
{
|
||||||
|
return $this->idle->count();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function getMaxConnections(): int
|
||||||
|
{
|
||||||
|
return $this->maxConnections;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return \Generator
|
||||||
|
*
|
||||||
|
* @resolve Link
|
||||||
|
*
|
||||||
|
* @throws FailureException If creating a new connection fails.
|
||||||
|
* @throws \Error If the pool has been closed.
|
||||||
|
*/
|
||||||
|
protected function pop(): \Generator
|
||||||
|
{
|
||||||
|
if ($this->closed) {
|
||||||
|
throw new \Error("The pool has been closed");
|
||||||
|
}
|
||||||
|
|
||||||
|
while ($this->promise !== null && $this->connections->count() + $this->pending >= $this->getMaxConnections()) {
|
||||||
|
yield $this->promise; // Prevent simultaneous connection creation when connection count is at maximum - 1.
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
|
// While loop to ensure an idle connection is available after promises below are resolved.
|
||||||
|
while ($this->idle->isEmpty()) {
|
||||||
|
if ($this->connections->count() + $this->pending < $this->getMaxConnections()) {
|
||||||
|
// Max connection count has not been reached, so open another connection.
|
||||||
|
++$this->pending;
|
||||||
|
try {
|
||||||
|
$connection = yield $this->connector->connect($this->connectionConfig);
|
||||||
|
if (!$connection instanceof Link) {
|
||||||
|
throw new \Error(\sprintf(
|
||||||
|
"%s::connect() must resolve to an instance of %s",
|
||||||
|
\get_class($this->connector),
|
||||||
|
Link::class
|
||||||
|
));
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
--$this->pending;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->connections->attach($connection);
|
||||||
|
return $connection;
|
||||||
|
}
|
||||||
|
|
||||||
|
// All possible connections busy, so wait until one becomes available.
|
||||||
|
try {
|
||||||
|
$this->deferred = new Deferred;
|
||||||
|
// May be resolved with defunct connection, but that connection will not be added to $this->idle.
|
||||||
|
yield $this->promise = $this->deferred->promise();
|
||||||
|
} finally {
|
||||||
|
$this->deferred = null;
|
||||||
|
$this->promise = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$connection = $this->idle->shift();
|
||||||
|
\assert($connection instanceof Link);
|
||||||
|
|
||||||
|
if ($connection->isAlive()) {
|
||||||
|
return $connection;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->connections->detach($connection);
|
||||||
|
} while (!$this->closed);
|
||||||
|
|
||||||
|
throw new FailureException("Pool closed before an active connection could be obtained");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Link $connection
|
||||||
|
*
|
||||||
|
* @throws \Error If the connection is not part of this pool.
|
||||||
|
*/
|
||||||
|
protected function push(Link $connection)
|
||||||
|
{
|
||||||
|
\assert(isset($this->connections[$connection]), 'Connection is not part of this pool');
|
||||||
|
|
||||||
|
if ($connection->isAlive()) {
|
||||||
|
$this->idle->push($connection);
|
||||||
|
} else {
|
||||||
|
$this->connections->detach($connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->deferred instanceof Deferred) {
|
||||||
|
$this->deferred->resolve($connection);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function query(string $sql): Promise
|
||||||
|
{
|
||||||
|
return call(function () use ($sql) {
|
||||||
|
$connection = yield from $this->pop();
|
||||||
|
\assert($connection instanceof Link);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$result = yield $connection->query($sql);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$this->push($connection);
|
||||||
|
throw $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($result instanceof Operation) {
|
||||||
|
$result->onDestruct(function () use ($connection) {
|
||||||
|
$this->push($connection);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
$this->push($connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function execute(string $sql, array $params = []): Promise
|
||||||
|
{
|
||||||
|
return call(function () use ($sql, $params) {
|
||||||
|
$connection = yield from $this->pop();
|
||||||
|
\assert($connection instanceof Link);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$result = yield $connection->execute($sql, $params);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$this->push($connection);
|
||||||
|
throw $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($result instanceof Operation) {
|
||||||
|
$result->onDestruct(function () use ($connection) {
|
||||||
|
$this->push($connection);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
$this->push($connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*
|
||||||
|
* Prepared statements returned by this method will stay alive as long as the pool remains open.
|
||||||
|
*/
|
||||||
|
public function prepare(string $sql): Promise
|
||||||
|
{
|
||||||
|
return call(function () use ($sql) {
|
||||||
|
$connection = yield from $this->pop();
|
||||||
|
\assert($connection instanceof Link);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$statement = yield $connection->prepare($sql);
|
||||||
|
\assert($statement instanceof Statement);
|
||||||
|
|
||||||
|
\assert(
|
||||||
|
$statement instanceof Operation,
|
||||||
|
Statement::class . " instances returned from connections must implement " . Operation::class
|
||||||
|
);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$this->push($connection);
|
||||||
|
throw $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
$statement->onDestruct(function () use ($connection) {
|
||||||
|
$this->push($connection);
|
||||||
|
});
|
||||||
|
|
||||||
|
return new PooledStatement($this, $statement, $this->prepare);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function transaction(int $isolation = Transaction::ISOLATION_COMMITTED): Promise
|
||||||
|
{
|
||||||
|
return call(function () use ($isolation) {
|
||||||
|
$connection = yield from $this->pop();
|
||||||
|
\assert($connection instanceof Link);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$transaction = yield $connection->transaction($isolation);
|
||||||
|
\assert($transaction instanceof Transaction);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$this->push($connection);
|
||||||
|
throw $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
$transaction->onDestruct(function () use ($connection) {
|
||||||
|
$this->push($connection);
|
||||||
|
});
|
||||||
|
|
||||||
|
return $transaction;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
145
src/PooledStatement.php
Normal file
145
src/PooledStatement.php
Normal file
@ -0,0 +1,145 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Amp\Sql;
|
||||||
|
|
||||||
|
use Amp\Loop;
|
||||||
|
use Amp\Promise;
|
||||||
|
use function Amp\call;
|
||||||
|
|
||||||
|
final class PooledStatement implements Statement
|
||||||
|
{
|
||||||
|
/** @var \Amp\Sql\Pool */
|
||||||
|
private $pool;
|
||||||
|
|
||||||
|
/** @var \SplQueue */
|
||||||
|
private $statements;
|
||||||
|
|
||||||
|
/** @var string */
|
||||||
|
private $sql;
|
||||||
|
|
||||||
|
/** @var int */
|
||||||
|
private $lastUsedAt;
|
||||||
|
|
||||||
|
/** @var string */
|
||||||
|
private $timeoutWatcher;
|
||||||
|
|
||||||
|
/** @var callable */
|
||||||
|
private $prepare;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Pool $pool Pool used to re-create the statement if the original closes.
|
||||||
|
* @param Statement $statement Original prepared statement returned from the Link.
|
||||||
|
* @param callable $prepare Callable that returns a new prepared statement.
|
||||||
|
*/
|
||||||
|
public function __construct(Pool $pool, Statement $statement, callable $prepare)
|
||||||
|
{
|
||||||
|
$this->lastUsedAt = \time();
|
||||||
|
$this->statements = $statements = new \SplQueue;
|
||||||
|
$this->pool = $pool;
|
||||||
|
$this->prepare = $prepare;
|
||||||
|
$this->sql = $statement->getQuery();
|
||||||
|
|
||||||
|
$this->statements->push($statement);
|
||||||
|
|
||||||
|
$this->timeoutWatcher = Loop::repeat(1000, static function () use ($pool, $statements) {
|
||||||
|
$now = \time();
|
||||||
|
$idleTimeout = ((int) ($pool->getIdleTimeout() / 10)) ?: 1;
|
||||||
|
|
||||||
|
while (!$statements->isEmpty()) {
|
||||||
|
/** @var Statement $statement */
|
||||||
|
$statement = $statements->bottom();
|
||||||
|
|
||||||
|
if ($statement->lastUsedAt() + $idleTimeout > $now) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$statements->shift();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Loop::unreference($this->timeoutWatcher);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __destruct()
|
||||||
|
{
|
||||||
|
Loop::cancel($this->timeoutWatcher);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*
|
||||||
|
* Unlike regular statements, as long as the pool is open this statement will not die.
|
||||||
|
*/
|
||||||
|
public function execute(array $params = []): Promise
|
||||||
|
{
|
||||||
|
$this->lastUsedAt = \time();
|
||||||
|
|
||||||
|
return call(function () use ($params) {
|
||||||
|
if (!$this->statements->isEmpty()) {
|
||||||
|
do {
|
||||||
|
/** @var Statement $statement */
|
||||||
|
$statement = $this->statements->shift();
|
||||||
|
} while (!$statement->isAlive() && !$this->statements->isEmpty());
|
||||||
|
} else {
|
||||||
|
$statement = yield ($this->prepare)($this->sql);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$result = yield $statement->execute($params);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$this->push($statement);
|
||||||
|
throw $exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($result instanceof Operation) {
|
||||||
|
$result->onDestruct(function () use ($statement) {
|
||||||
|
$this->push($statement);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
$this->push($statement);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Only retains statements if less than 10% of the pool is consumed by this statement and the pool has
|
||||||
|
* available connections.
|
||||||
|
*
|
||||||
|
* @param Statement $statement
|
||||||
|
*/
|
||||||
|
private function push(Statement $statement)
|
||||||
|
{
|
||||||
|
$maxConnections = $this->pool->getMaxConnections();
|
||||||
|
|
||||||
|
if ($this->statements->count() > ($maxConnections / 10)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($maxConnections === $this->pool->getConnectionCount() && $this->pool->getIdleConnectionCount() === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->statements->push($statement);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/** {@inheritdoc} */
|
||||||
|
public function isAlive(): bool
|
||||||
|
{
|
||||||
|
return $this->pool->isAlive();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** {@inheritdoc} */
|
||||||
|
public function getQuery(): string
|
||||||
|
{
|
||||||
|
return $this->sql;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** {@inheritdoc} */
|
||||||
|
public function lastUsedAt(): int
|
||||||
|
{
|
||||||
|
return $this->lastUsedAt;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user