Compare commits
19 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ccbac52a16 | |||
| 9fe0698d1c | |||
| 55e1f6235e | |||
| ed0d044223 | |||
| 2b85cb4ec3 | |||
| 73b9923740 | |||
| b2c8160314 | |||
| 199312d326 | |||
| 87f568bdfe | |||
| 2fd37f90a3 | |||
| c100190155 | |||
| 71766ee914 | |||
| 3a39eaabf4 | |||
| 722b286f91 | |||
| f9ee3aa014 | |||
| 4341efcb8c | |||
| 8acf74c9ed | |||
| b9750d743b | |||
| ea34371652 |
+1
-1
@@ -204,7 +204,7 @@ class ActiveQuery extends QueryTrait implements ISqlBuilder
|
|||||||
*/
|
*/
|
||||||
public function exists(): bool
|
public function exists(): bool
|
||||||
{
|
{
|
||||||
return $this->buildCommand($this->builder->one())->rowCount() > 0;
|
return $this->buildCommand($this->limit(1)->builder->exists())->exists();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Database\Base;
|
||||||
|
|
||||||
|
class PDO extends \PDO
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $database
|
||||||
|
* @param string $host
|
||||||
|
* @param string $username
|
||||||
|
* @param string $password
|
||||||
|
* @param array $options
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
readonly public string $database,
|
||||||
|
readonly public string $host,
|
||||||
|
readonly public string $username,
|
||||||
|
readonly public string $password,
|
||||||
|
readonly public array $options,
|
||||||
|
)
|
||||||
|
{
|
||||||
|
parent::__construct('mysql:dbname=' . $this->database . ';host=' . $this->host, $this->username, $this->password, $this->options);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
+16
-39
@@ -75,6 +75,19 @@ class Command extends Component
|
|||||||
return $this->search('fetch');
|
return $this->search('fetch');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return bool
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function exists(): bool
|
||||||
|
{
|
||||||
|
$data = $this->search('fetch');
|
||||||
|
if (!$data) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return mixed
|
* @return mixed
|
||||||
* @throws
|
* @throws
|
||||||
@@ -88,46 +101,11 @@ class Command extends Component
|
|||||||
* @return mixed
|
* @return mixed
|
||||||
* @throws
|
* @throws
|
||||||
*/
|
*/
|
||||||
public function rowCount(): mixed
|
public function rowCount(): int
|
||||||
{
|
{
|
||||||
$client = $this->connection->getConnection();
|
$data = $this->search('fetch');
|
||||||
try {
|
|
||||||
$client->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, true);
|
|
||||||
if (($prepare = $client->query($this->sql)) === false) {
|
|
||||||
throw new Exception('(' . $prepare->errorInfo()[0] . ')' . $client->errorInfo()[2]);
|
|
||||||
}
|
|
||||||
|
|
||||||
$prepare->execute($this->params);
|
return !$data ? 0 : +current($data);
|
||||||
|
|
||||||
$count = $prepare->rowCount();
|
|
||||||
$prepare->closeCursor();
|
|
||||||
|
|
||||||
$this->connection->println($this->sql, $this->params);
|
|
||||||
|
|
||||||
return $count;
|
|
||||||
} catch (Throwable $throwable) {
|
|
||||||
if ($this->isRefresh($throwable)) return $this->rowCount();
|
|
||||||
|
|
||||||
$errorMsg = $throwable->getMessage() . PHP_EOL . ' Sql: ' . $this->sql . '.' . json_encode($this->params);
|
|
||||||
$this->getLogger()->failure($errorMsg . PHP_EOL, 'mysql');
|
|
||||||
return 0;
|
|
||||||
} finally {
|
|
||||||
$this->connection->release($client);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return bool
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public function exists(): bool
|
|
||||||
{
|
|
||||||
$total = $this->search('rowCount');
|
|
||||||
if ($total === false) {
|
|
||||||
throw new Exception('Query data is has error.');
|
|
||||||
}
|
|
||||||
return $total > 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -182,7 +160,6 @@ class Command extends Component
|
|||||||
{
|
{
|
||||||
$client = $this->connection->getConnection();
|
$client = $this->connection->getConnection();
|
||||||
try {
|
try {
|
||||||
$this->connection->println($this->sql, $this->params);
|
|
||||||
if (($prepare = $client->prepare($this->sql)) === false) {
|
if (($prepare = $client->prepare($this->sql)) === false) {
|
||||||
throw new Exception('(' . $prepare->errorInfo()[0] . ')' . $prepare->errorInfo()[2]);
|
throw new Exception('(' . $prepare->errorInfo()[0] . ')' . $prepare->errorInfo()[2]);
|
||||||
}
|
}
|
||||||
|
|||||||
+52
-39
@@ -21,7 +21,6 @@ use Kiri\Abstracts\Component;
|
|||||||
use Kiri\Di\Context;
|
use Kiri\Di\Context;
|
||||||
use Kiri\Pool\Pool;
|
use Kiri\Pool\Pool;
|
||||||
use Kiri\Events\EventProvider;
|
use Kiri\Events\EventProvider;
|
||||||
use PDO;
|
|
||||||
use Kiri\Error\StdoutLogger;
|
use Kiri\Error\StdoutLogger;
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
use Kiri\Server\Events\OnWorkerStart;
|
use Kiri\Server\Events\OnWorkerStart;
|
||||||
@@ -29,6 +28,9 @@ use Kiri\Server\Events\OnTaskerStart;
|
|||||||
use Kiri\Server\Events\OnAfterRequest;
|
use Kiri\Server\Events\OnAfterRequest;
|
||||||
use Kiri\Di\Inject\Container;
|
use Kiri\Di\Inject\Container;
|
||||||
use Swoole\Timer;
|
use Swoole\Timer;
|
||||||
|
use Database\Base\PDO;
|
||||||
|
|
||||||
|
//use PDO;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class Connection
|
* Class Connection
|
||||||
@@ -37,25 +39,26 @@ use Swoole\Timer;
|
|||||||
class Connection extends Component
|
class Connection extends Component
|
||||||
{
|
{
|
||||||
|
|
||||||
public string $id = 'db';
|
public string $id = 'db';
|
||||||
public string $cds = '';
|
public string $cds = '';
|
||||||
public string $password = '';
|
public string $password = '';
|
||||||
public string $username = '';
|
public string $username = '';
|
||||||
public string $charset = 'utf-8';
|
public string $charset = 'utf-8';
|
||||||
public string $tablePrefix = '';
|
public string $tablePrefix = '';
|
||||||
public string $database = '';
|
public string $database = '';
|
||||||
public int $timeout = 30;
|
public int $timeout = 30;
|
||||||
public int $waite_time = 3;
|
public int $waite_time = 3;
|
||||||
public int $tick_time = 60;
|
public int $tick_time = 60;
|
||||||
public int $idle_count = 3;
|
public int $idle_count = 3;
|
||||||
public int $idle_time = 60;
|
public int $idle_time = 60;
|
||||||
public array $pool = ['max' => 10, 'min' => 1];
|
public array $pool = ['max' => 10, 'min' => 1];
|
||||||
private int $storey = 0;
|
private int $storey = 0;
|
||||||
protected int $timerId = -1;
|
protected int $timerId = -1;
|
||||||
public bool $enableCache = false;
|
public bool $enableCache = false;
|
||||||
public string $cacheDriver = 'redis';
|
public string $cacheDriver = 'redis';
|
||||||
public array $attributes = [];
|
public array $attributes = [];
|
||||||
protected \Closure $_println;
|
public array $slave = [];
|
||||||
|
protected ?\Closure $_println = null;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -121,14 +124,14 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
protected function checkClientHealth(Pool $pool): void
|
protected function checkClientHealth(Pool $pool): void
|
||||||
{
|
{
|
||||||
$pool->flush($this->cds, $this->pool['min'] ?? 1);
|
$pool->flush($this->getName(), $this->pool['min'] ?? 1);
|
||||||
$length = $pool->size($this->cds);
|
$length = $pool->size($this->getName());
|
||||||
for ($i = 0; $i < $length; $i++) {
|
for ($i = 0; $i < $length; $i++) {
|
||||||
try {
|
try {
|
||||||
if (($client = $this->validator($pool)) === false) {
|
if (($client = $this->validator($pool)) === false) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
$pool->push($this->cds, $client);
|
$pool->push($this->getName(), $client);
|
||||||
} catch (\Throwable $exception) {
|
} catch (\Throwable $exception) {
|
||||||
if (!str_contains($exception->getMessage(), 'Client timeout.')) {
|
if (!str_contains($exception->getMessage(), 'Client timeout.')) {
|
||||||
$this->logger->error(throwable($exception), [$this->cds]);
|
$this->logger->error(throwable($exception), [$this->cds]);
|
||||||
@@ -138,6 +141,15 @@ class Connection extends Component
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private function getName(): string
|
||||||
|
{
|
||||||
|
return 'mysql.' . $this->cds;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param Pool $pool
|
* @param Pool $pool
|
||||||
* @return PDO|bool
|
* @return PDO|bool
|
||||||
@@ -146,7 +158,7 @@ class Connection extends Component
|
|||||||
protected function validator(Pool $pool): PDO|bool
|
protected function validator(Pool $pool): PDO|bool
|
||||||
{
|
{
|
||||||
/** @var $client PDO */
|
/** @var $client PDO */
|
||||||
if (($client = $pool->get($this->cds)) === false) {
|
if (($client = $pool->get($this->getName())) === false) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if ($client->query('select 1') === false) {
|
if ($client->query('select 1') === false) {
|
||||||
@@ -176,7 +188,7 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
protected function getNormalClientHealth(): PDO
|
protected function getNormalClientHealth(): PDO
|
||||||
{
|
{
|
||||||
$data = $this->pool()->get($this->cds, $this->waite_time);
|
$data = $this->pool()->get($this->getName(), $this->waite_time);
|
||||||
if ($data === false) {
|
if ($data === false) {
|
||||||
throw new Exception('Client Waite timeout.');
|
throw new Exception('Client Waite timeout.');
|
||||||
}
|
}
|
||||||
@@ -302,7 +314,7 @@ class Connection extends Component
|
|||||||
public function release(PDO $pdo): void
|
public function release(PDO $pdo): void
|
||||||
{
|
{
|
||||||
if (!$this->inTransaction()) {
|
if (!$this->inTransaction()) {
|
||||||
$this->pool()->push($this->cds, $pdo);
|
$this->pool()->push($this->getName(), $pdo);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -314,7 +326,7 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
public function clear_connection(): void
|
public function clear_connection(): void
|
||||||
{
|
{
|
||||||
$this->pool()->flush($this->cds, 0);
|
$this->pool()->flush($this->getName(), 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -326,23 +338,24 @@ class Connection extends Component
|
|||||||
if ($this->timerId > -1) {
|
if ($this->timerId > -1) {
|
||||||
Timer::clear($this->timerId);
|
Timer::clear($this->timerId);
|
||||||
}
|
}
|
||||||
$this->pool()->close($this->cds);
|
$this->pool()->close($this->getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return PDO
|
* @return PDO
|
||||||
*/
|
*/
|
||||||
public function newConnect(): PDO
|
public function newConnect(): \PDO
|
||||||
{
|
{
|
||||||
$pdo = new PDO('mysql:dbname=' . $this->database . ';host=' . $this->cds, $this->username, $this->password, [
|
$pdo = new PDO($this->database, $this->cds, $this->username, $this->password, [
|
||||||
PDO::ATTR_CASE => PDO::CASE_NATURAL,
|
// $pdo = new \PDO('mysql:dbname=' . $this->database . ';host=' . $this->cds, $this->username, $this->password, [
|
||||||
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
|
\PDO::ATTR_CASE => \PDO::CASE_NATURAL,
|
||||||
PDO::ATTR_ORACLE_NULLS => PDO::NULL_NATURAL,
|
\PDO::ATTR_ERRMODE => \PDO::ERRMODE_EXCEPTION,
|
||||||
PDO::ATTR_STRINGIFY_FETCHES => false,
|
\PDO::ATTR_ORACLE_NULLS => \PDO::NULL_NATURAL,
|
||||||
PDO::ATTR_EMULATE_PREPARES => true,
|
\PDO::ATTR_STRINGIFY_FETCHES => false,
|
||||||
PDO::ATTR_TIMEOUT => $this->timeout,
|
\PDO::ATTR_EMULATE_PREPARES => true,
|
||||||
PDO::MYSQL_ATTR_INIT_COMMAND => 'SET NAMES ' . $this->charset
|
\PDO::ATTR_TIMEOUT => $this->timeout,
|
||||||
|
\PDO::MYSQL_ATTR_INIT_COMMAND => 'SET NAMES ' . $this->charset
|
||||||
]);
|
]);
|
||||||
foreach ($this->attributes as $key => $attribute) {
|
foreach ($this->attributes as $key => $attribute) {
|
||||||
$pdo->setAttribute($key, $attribute);
|
$pdo->setAttribute($key, $attribute);
|
||||||
@@ -356,8 +369,8 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
protected function pool(): Pool
|
protected function pool(): Pool
|
||||||
{
|
{
|
||||||
if (!$this->connections->hasChannel($this->cds)) {
|
if (!$this->connections->hasChannel($this->getName())) {
|
||||||
$this->connections->created($this->cds, $this->pool['max'] ?? 1, [$this, 'newConnect']);
|
$this->connections->created($this->getName(), $this->pool['max'] ?? 1, [$this, 'newConnect']);
|
||||||
}
|
}
|
||||||
return $this->connections;
|
return $this->connections;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -115,10 +115,11 @@ class Db extends QueryTrait implements ISqlBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @return bool
|
* @return bool
|
||||||
|
* @throws Exception
|
||||||
*/
|
*/
|
||||||
public function exists(): bool
|
public function exists(): bool
|
||||||
{
|
{
|
||||||
return $this->connection->createCommand(SqlBuilder::builder($this)->one())->rowCount() > 0;
|
return $this->connection->createCommand(SqlBuilder::builder($this->limit(1))->exists())->exists();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -151,6 +152,55 @@ class Db extends QueryTrait implements ISqlBuilder
|
|||||||
return $this->connection->createCommand(SqlBuilder::builder($this)->delete())->delete();
|
return $this->connection->createCommand(SqlBuilder::builder($this)->delete())->delete();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return bool|array|null
|
||||||
|
*/
|
||||||
|
public function first(): bool|array|null
|
||||||
|
{
|
||||||
|
return $this->connection->createCommand(SqlBuilder::builder($this)->one())->one();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return bool|array
|
||||||
|
*/
|
||||||
|
public function get(): bool|array
|
||||||
|
{
|
||||||
|
return $this->connection->createCommand(SqlBuilder::builder($this)->all())->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $sql
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function exec(string $sql): mixed
|
||||||
|
{
|
||||||
|
return $this->connection->createCommand($sql)->exec();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $sql
|
||||||
|
* @return array|bool|null
|
||||||
|
*/
|
||||||
|
public function query(string $sql): array|bool|null
|
||||||
|
{
|
||||||
|
return $this->connection->createCommand($sql)->one();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $sql
|
||||||
|
* @return array|bool
|
||||||
|
*/
|
||||||
|
public function queryAll(string $sql): array|bool
|
||||||
|
{
|
||||||
|
return $this->connection->createCommand($sql)->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string $table
|
* @param string $table
|
||||||
* @return array|bool|null
|
* @return array|bool|null
|
||||||
|
|||||||
+11
-1
@@ -235,7 +235,17 @@ class SqlBuilder extends Component
|
|||||||
*/
|
*/
|
||||||
public function count(): string
|
public function count(): string
|
||||||
{
|
{
|
||||||
return $this->makeSelect() . $this->make();
|
return $this->makeSelect(['COUNT(*)']) . $this->make();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
* @throws
|
||||||
|
*/
|
||||||
|
public function exists(): string
|
||||||
|
{
|
||||||
|
return $this->makeSelect(['0']) . $this->make();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user