diff --git a/src/Connection.php b/src/Connection.php index e4fae4c..9d0dc3f 100644 --- a/src/Connection.php +++ b/src/Connection.php @@ -33,31 +33,35 @@ interface Connection /** * Create new connection. * - * @param Driver $driver - * @param Credentials $credentials - * @param AbstractPlatform $platform + * @param Driver $driver + * @param Credentials $credentials + * @param AbstractPlatform $platform + * @param ConnectionOptions|null $options * * @return Connection */ public static function create( Driver $driver, Credentials $credentials, - AbstractPlatform $platform + AbstractPlatform $platform, + ?ConnectionOptions $options = null ): Connection; /** * Create new connection. * - * @param Driver $driver - * @param Credentials $credentials - * @param AbstractPlatform $platform + * @param Driver $driver + * @param Credentials $credentials + * @param AbstractPlatform $platform + * @param ConnectionOptions|null $options * * @return Connection */ public static function createConnected( Driver $driver, Credentials $credentials, - AbstractPlatform $platform + AbstractPlatform $platform, + ?ConnectionOptions $options = null ): Connection; /** @@ -67,8 +71,10 @@ public function getDriverNamespace(): string; /** * Connect. + * + * @param ConnectionOptions|null $options */ - public function connect(); + public function connect(?ConnectionOptions $options = null); /** * Close. diff --git a/src/ConnectionOptions.php b/src/ConnectionOptions.php new file mode 100644 index 0000000..cee7635 --- /dev/null +++ b/src/ConnectionOptions.php @@ -0,0 +1,34 @@ +keepAliveIntervalSec = $keepAliveIntervalSec; + } + + /** + * @return int + */ + public function getKeepAliveIntervalSec(): int + { + return $this->keepAliveIntervalSec; + } + + /** + * @param int $keepAliveIntervalSec + */ + public function setKeepAliveIntervalSec(int $keepAliveIntervalSec): void + { + $this->keepAliveIntervalSec = $keepAliveIntervalSec; + } +} diff --git a/src/ConnectionPool.php b/src/ConnectionPool.php index 466a57b..149cc5b 100644 --- a/src/ConnectionPool.php +++ b/src/ConnectionPool.php @@ -45,23 +45,31 @@ private function __construct(array $connections) /** * Create new connection. * - * @param Driver $driver - * @param Credentials $credentials - * @param AbstractPlatform $platform + * @param Driver $driver + * @param Credentials $credentials + * @param AbstractPlatform $platform + * @param ConnectionOptions|null $options * * @return Connection */ public static function create( Driver $driver, Credentials $credentials, - AbstractPlatform $platform + AbstractPlatform $platform, + ?ConnectionOptions $options = null ): Connection { - $numberOfConnections = $credentials->getConnections(); + $numberOfConnections = $credentials->getConnections() ?? + ConnectionPoolOptions::DEFAULT_NUMBER_OF_CONNECTIONS; + if ($options instanceof ConnectionPoolOptions) { + $numberOfConnections = $options->getNumberOfConnections(); + } + if ($numberOfConnections <= 1) { return SingleConnection::create( $driver, $credentials, - $platform + $platform, + $options ); } @@ -71,7 +79,8 @@ public static function create( SingleConnection::create( clone $driver, $credentials, - $platform + $platform, + $options ), $i ); } @@ -82,19 +91,21 @@ public static function create( /** * Create new connection. * - * @param Driver $driver - * @param Credentials $credentials - * @param AbstractPlatform $platform + * @param Driver $driver + * @param Credentials $credentials + * @param AbstractPlatform $platform + * @param ConnectionOptions|null $options * * @return Connection */ public static function createConnected( Driver $driver, Credentials $credentials, - AbstractPlatform $platform + AbstractPlatform $platform, + ?ConnectionOptions $options = null ): Connection { - $connection = self::create($driver, $credentials, $platform); - $connection->connect(); + $connection = self::create($driver, $credentials, $platform, $options); + $connection->connect($options); return $connection; } @@ -161,10 +172,10 @@ private function executeInBestConnection(callable $callable): PromiseInterface /** * Connect. */ - public function connect() + public function connect(?ConnectionOptions $options = null) { foreach ($this->connections as $connection) { - $connection->getConnection()->connect(); + $connection->getConnection()->connect($options); } } diff --git a/src/ConnectionPoolOptions.php b/src/ConnectionPoolOptions.php new file mode 100644 index 0000000..fcdbc13 --- /dev/null +++ b/src/ConnectionPoolOptions.php @@ -0,0 +1,38 @@ +numberOfConnections = $numberOfConnections; + parent::__construct($keepAliveIntervalSec); + } + + /** + * @return int + */ + public function getNumberOfConnections(): int + { + return $this->numberOfConnections; + } + + /** + * @param int $numberOfConnections + */ + public function setNumberOfConnections(int $numberOfConnections): void + { + $this->numberOfConnections = $numberOfConnections; + } +} diff --git a/src/Credentials.php b/src/Credentials.php index dcd9aec..40d2320 100644 --- a/src/Credentials.php +++ b/src/Credentials.php @@ -25,8 +25,18 @@ class Credentials private string $user; private string $password; private string $dbName; + + /** + * @deprecated + * @var array options + */ private array $options; - private int $connections; + + /** + * @deprecated + * @var int|null $connections + */ + private ?int $connections = null; /** * Credentials constructor. @@ -36,25 +46,36 @@ class Credentials * @param string $user * @param string $password * @param string $dbName - * @param array $options - * @param int $connections */ public function __construct( string $host, string $port, string $user, string $password, - string $dbName, - array $options = [], - int $connections = 1 + string $dbName ) { $this->host = $host; $this->port = $port; $this->user = $user; $this->password = $password; $this->dbName = $dbName; - $this->options = $options; - $this->connections = $connections; + + if (func_num_args() > 5) { + trigger_error( + '6th argument is deprecated, for options please use ' . ConnectionOptions::class . + ' or and extend of this class instead, when creating a connection', + E_USER_DEPRECATED + ); + $this->options = (array)func_get_arg(5); + } + if (func_num_args() > 6) { + trigger_error( + '7th argument is deprecated, please use ' . ConnectionPoolOptions::class . + ' to set the number of connections, when creating a ' . ConnectionPool::class, + E_USER_DEPRECATED + ); + $this->connections = (int)func_get_arg(6); + } } /** @@ -98,6 +119,7 @@ public function getDbName(): string } /** + * @deprecated * @return array */ public function getOptions(): array @@ -106,9 +128,10 @@ public function getOptions(): array } /** - * @return int + * @deprecated + * @return int|null */ - public function getConnections(): int + public function getConnections(): ?int { return $this->connections; } diff --git a/src/SingleConnection.php b/src/SingleConnection.php index 6b113f9..82b4205 100644 --- a/src/SingleConnection.php +++ b/src/SingleConnection.php @@ -25,6 +25,8 @@ use Drift\DBAL\Driver\Driver; use Drift\DBAL\Mock\MockedDBALConnection; use Drift\DBAL\Mock\MockedDriver; +use React\EventLoop\Loop; +use React\EventLoop\TimerInterface; use function React\Promise\map; use React\Promise\PromiseInterface; @@ -33,6 +35,7 @@ class SingleConnection implements Connection private Driver $driver; private Credentials $credentials; private AbstractPlatform $platform; + private ?TimerInterface $keepAliveTimer = null; /** * Connection constructor. @@ -51,19 +54,28 @@ private function __construct( $this->platform = $platform; } + public function __destruct() + { + if ($this->keepAliveTimer != null) { + Loop::get()->cancelTimer($this->keepAliveTimer); + } + } + /** * Create new connection. * - * @param Driver $driver - * @param Credentials $credentials - * @param AbstractPlatform $platform + * @param Driver $driver + * @param Credentials $credentials + * @param AbstractPlatform $platform + * @param ConnectionOptions|null $options * * @return Connection */ public static function create( Driver $driver, Credentials $credentials, - AbstractPlatform $platform + AbstractPlatform $platform, + ?ConnectionOptions $options = null ): Connection { return new self($driver, $credentials, $platform); } @@ -71,19 +83,21 @@ public static function create( /** * Create new connection. * - * @param Driver $driver - * @param Credentials $credentials - * @param AbstractPlatform $platform + * @param Driver $driver + * @param Credentials $credentials + * @param AbstractPlatform $platform + * @param ConnectionOptions|null $options * * @return Connection */ public static function createConnected( Driver $driver, Credentials $credentials, - AbstractPlatform $platform + AbstractPlatform $platform, + ?ConnectionOptions $options = null ): Connection { $connection = self::create($driver, $credentials, $platform); - $connection->connect(); + $connection->connect($options); return $connection; } @@ -99,11 +113,21 @@ public function getDriverNamespace(): string /** * Connect. */ - public function connect() + public function connect(?ConnectionOptions $options = null) { $this ->driver ->connect($this->credentials); + + if ($options != null && $options->getKeepAliveIntervalSec() > 0) { + $this->keepAliveTimer = Loop::get()->addPeriodicTimer($options->getKeepAliveIntervalSec(), function() { + $qb = $this->createQueryBuilder(); + $this->query( + $qb + ->select('1') + ); + }); + } } /** @@ -111,6 +135,11 @@ public function connect() */ public function close() { + if ($this->keepAliveTimer != null) { + Loop::get()->cancelTimer($this->keepAliveTimer); + $this->keepAliveTimer = null; + } + $this ->driver ->close(); diff --git a/tests/Mysql5ConnectionPoolTest.php b/tests/Mysql5ConnectionPoolTest.php index e8816ac..ccc89f2 100644 --- a/tests/Mysql5ConnectionPoolTest.php +++ b/tests/Mysql5ConnectionPoolTest.php @@ -18,6 +18,7 @@ use Doctrine\DBAL\Platforms\MySQLPlatform; use Drift\DBAL\Connection; use Drift\DBAL\ConnectionPool; +use Drift\DBAL\ConnectionPoolOptions; use Drift\DBAL\Credentials; use Drift\DBAL\Driver\Mysql\MysqlDriver; use React\EventLoop\LoopInterface; @@ -41,8 +42,7 @@ protected function getConnection(LoopInterface $loop): Connection '3306', 'root', 'root', - 'test', - [], 10 - ), $mysqlPlatform); + 'test' + ), $mysqlPlatform, new ConnectionPoolOptions(10)); } } diff --git a/tests/PostgreSQLConnectionPoolTest.php b/tests/PostgreSQLConnectionPoolTest.php index a2472db..c2ff308 100644 --- a/tests/PostgreSQLConnectionPoolTest.php +++ b/tests/PostgreSQLConnectionPoolTest.php @@ -18,6 +18,7 @@ use Doctrine\DBAL\Platforms\PostgreSQLPlatform; use Drift\DBAL\Connection; use Drift\DBAL\ConnectionPool; +use Drift\DBAL\ConnectionPoolOptions; use Drift\DBAL\Credentials; use Drift\DBAL\Driver\PostgreSQL\PostgreSQLDriver; use React\EventLoop\LoopInterface; @@ -40,7 +41,6 @@ protected function getConnection(LoopInterface $loop): Connection 'root', 'root', 'test', - [], 10 - ), $postgreSQLPlatform); + ), $postgreSQLPlatform, new ConnectionPoolOptions(10)); } } diff --git a/tests/PostgreSQLConnectionTest.php b/tests/PostgreSQLConnectionTest.php index e5e92ad..001c544 100644 --- a/tests/PostgreSQLConnectionTest.php +++ b/tests/PostgreSQLConnectionTest.php @@ -17,6 +17,7 @@ use Doctrine\DBAL\Platforms\PostgreSQLPlatform; use Drift\DBAL\Connection; +use Drift\DBAL\ConnectionPoolOptions; use Drift\DBAL\Credentials; use Drift\DBAL\Driver\PostgreSQL\PostgreSQLDriver; use Drift\DBAL\SingleConnection; @@ -40,6 +41,6 @@ public function getConnection(LoopInterface $loop): Connection 'root', 'root', 'test' - ), $postgreSQLPlatform); + ), $postgreSQLPlatform, new ConnectionPoolOptions(10)); } }