eee
This commit is contained in:
+23
-8
@@ -30,6 +30,14 @@ use ReturnTypeWillChange;
|
|||||||
use ReflectionException;
|
use ReflectionException;
|
||||||
use validator\Validator;
|
use validator\Validator;
|
||||||
|
|
||||||
|
|
||||||
|
enum Driver
|
||||||
|
{
|
||||||
|
case Mysql;
|
||||||
|
case Pgsql;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class BOrm
|
* Class BOrm
|
||||||
*
|
*
|
||||||
@@ -139,7 +147,7 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
*/
|
*/
|
||||||
public function clean(): void
|
public function clean(): void
|
||||||
{
|
{
|
||||||
$this->_attributes = [];
|
$this->_attributes = [];
|
||||||
$this->_oldAttributes = [];
|
$this->_oldAttributes = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -449,7 +457,7 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
*/
|
*/
|
||||||
private function insert(): bool|static
|
private function insert(): bool|static
|
||||||
{
|
{
|
||||||
$sql = SqlBuilder::builder($query = static::query())->insert($this->_attributes);
|
$sql = SqlBuilder::builder($query = static::query())->insert($this->_attributes);
|
||||||
$lastId = $this->getConnection()->createCommand($sql, $query->params)->exec();
|
$lastId = $this->getConnection()->createCommand($sql, $query->params)->exec();
|
||||||
if ($lastId === FALSE) {
|
if ($lastId === FALSE) {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@@ -512,7 +520,7 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
protected function arrayIntersect(array $params): array
|
protected function arrayIntersect(array $params): array
|
||||||
{
|
{
|
||||||
$condition = [];
|
$condition = [];
|
||||||
$oldPrams = [];
|
$oldPrams = [];
|
||||||
foreach ($this->_oldAttributes as $key => $attribute) {
|
foreach ($this->_oldAttributes as $key => $attribute) {
|
||||||
if (!array_key_exists($key, $params) || $params[$key] == $attribute) {
|
if (!array_key_exists($key, $params) || $params[$key] == $attribute) {
|
||||||
$condition[$key] = $attribute;
|
$condition[$key] = $attribute;
|
||||||
@@ -547,7 +555,7 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
*/
|
*/
|
||||||
public function populates(array $value): static
|
public function populates(array $value): static
|
||||||
{
|
{
|
||||||
$this->_attributes = $value;
|
$this->_attributes = $value;
|
||||||
$this->_oldAttributes = $value;
|
$this->_oldAttributes = $value;
|
||||||
$this->setIsNowExample();
|
$this->setIsNowExample();
|
||||||
return $this;
|
return $this;
|
||||||
@@ -635,7 +643,7 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
*/
|
*/
|
||||||
public function getTable(): string
|
public function getTable(): string
|
||||||
{
|
{
|
||||||
$connection = $this->getConnection();
|
$connection = $this->getConnection();
|
||||||
$tablePrefix = $connection->tablePrefix;
|
$tablePrefix = $connection->tablePrefix;
|
||||||
if (empty($this->table)) {
|
if (empty($this->table)) {
|
||||||
throw new Exception('You need add static method `tableName` and return table name.');
|
throw new Exception('You need add static method `tableName` and return table name.');
|
||||||
@@ -644,6 +652,13 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
if (!empty($tablePrefix) && !str_starts_with($table, $tablePrefix)) {
|
if (!empty($tablePrefix) && !str_starts_with($table, $tablePrefix)) {
|
||||||
$table = $tablePrefix . $table;
|
$table = $tablePrefix . $table;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$driver = strtolower($connection->driver ?? 'mysql');
|
||||||
|
if (in_array($driver, ['pgsql', 'postgresql'])) {
|
||||||
|
// PostgreSQL 使用双引号,并且 schema.table 格式
|
||||||
|
return '"' . $connection->database . '"."' . $table . '"';
|
||||||
|
}
|
||||||
|
// MySQL 使用反引号
|
||||||
return '`' . $connection->database . '`.' . $table;
|
return '`' . $connection->database . '`.' . $table;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -678,7 +693,7 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
public function refresh(): static
|
public function refresh(): static
|
||||||
{
|
{
|
||||||
$this->_oldAttributes = $this->_attributes;
|
$this->_oldAttributes = $this->_attributes;
|
||||||
$this->isNewExample = FALSE;
|
$this->isNewExample = FALSE;
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -853,8 +868,8 @@ abstract class Model extends Component implements ModelInterface, ArrayAccess, \
|
|||||||
*/
|
*/
|
||||||
public static function populate(array $data): static
|
public static function populate(array $data): static
|
||||||
{
|
{
|
||||||
$model = new static();
|
$model = new static();
|
||||||
$model->_attributes = $data;
|
$model->_attributes = $data;
|
||||||
$model->_oldAttributes = $data;
|
$model->_oldAttributes = $data;
|
||||||
$model->setIsNowExample();
|
$model->setIsNowExample();
|
||||||
return $model;
|
return $model;
|
||||||
|
|||||||
+16
-1
@@ -5,6 +5,7 @@ namespace Database\Base;
|
|||||||
class PDO extends \PDO
|
class PDO extends \PDO
|
||||||
{
|
{
|
||||||
|
|
||||||
|
readonly public string $driver;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string $database
|
* @param string $database
|
||||||
@@ -12,6 +13,7 @@ class PDO extends \PDO
|
|||||||
* @param string $username
|
* @param string $username
|
||||||
* @param string $password
|
* @param string $password
|
||||||
* @param array $options
|
* @param array $options
|
||||||
|
* @param string $driver
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
readonly public string $database,
|
readonly public string $database,
|
||||||
@@ -19,10 +21,23 @@ class PDO extends \PDO
|
|||||||
readonly public string $username,
|
readonly public string $username,
|
||||||
readonly public string $password,
|
readonly public string $password,
|
||||||
readonly public array $options,
|
readonly public array $options,
|
||||||
|
string $driver = 'mysql',
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
parent::__construct('mysql:dbname=' . $this->database . ';host=' . $this->host, $this->username, $this->password, $this->options);
|
$this->driver = strtolower($driver);
|
||||||
|
$dsn = $this->buildDsn();
|
||||||
|
parent::__construct($dsn, $this->username, $this->password, $this->options);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private function buildDsn(): string
|
||||||
|
{
|
||||||
|
return match ($this->driver) {
|
||||||
|
'pgsql', 'postgresql' => 'pgsql:host=' . $this->host . ';dbname=' . $this->database,
|
||||||
|
default => 'mysql:dbname=' . $this->database . ';host=' . $this->host,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
+21
-3
@@ -188,15 +188,33 @@ class Command extends Component
|
|||||||
*/
|
*/
|
||||||
protected function isRefresh(Throwable $throwable): bool
|
protected function isRefresh(Throwable $throwable): bool
|
||||||
{
|
{
|
||||||
if (str_contains($throwable->getMessage(), 'MySQL server has gone away')) {
|
$message = $throwable->getMessage();
|
||||||
|
|
||||||
|
// MySQL 错误处理
|
||||||
|
if (str_contains($message, 'MySQL server has gone away')) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (str_contains($throwable->getMessage(), 'Send of 14 bytes failed with errno=32 Broken pipe')) {
|
if (str_contains($message, 'Send of 14 bytes failed with errno=32 Broken pipe')) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (str_contains($throwable->getMessage(), 'Lost connection to MySQL server during query')) {
|
if (str_contains($message, 'Lost connection to MySQL server during query')) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// PostgreSQL 错误处理
|
||||||
|
if (str_contains($message, 'server closed the connection unexpectedly')) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (str_contains($message, 'Connection refused')) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (str_contains($message, 'Broken pipe')) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (str_contains($message, 'connection to server was lost')) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -43,10 +43,9 @@ abstract class Condition extends Component
|
|||||||
public function setValue($params): void
|
public function setValue($params): void
|
||||||
{
|
{
|
||||||
if (is_array($params)) {
|
if (is_array($params)) {
|
||||||
$values = [];
|
$values = array_map(function ($value) {
|
||||||
foreach ($params as $item => $value) {
|
return is_numeric($value) ? $value : '\'' . $value . '\'';
|
||||||
$values[$item] = is_numeric($value) ? $value : '\'' . $value . '\'';
|
}, $params);
|
||||||
}
|
|
||||||
$this->value = $values;
|
$this->value = $values;
|
||||||
} else {
|
} else {
|
||||||
$this->value = $this->checkIsSqlString($params);
|
$this->value = $this->checkIsSqlString($params);
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ class NotInCondition extends Condition
|
|||||||
* @return string|null
|
* @return string|null
|
||||||
* @throws
|
* @throws
|
||||||
*/
|
*/
|
||||||
#[Pure] public function builder(): ?string
|
public function builder(): ?string
|
||||||
{
|
{
|
||||||
if (!is_array($this->value)) {
|
if (!is_array($this->value)) {
|
||||||
throw new \Exception('Builder data by a empty string. need array');
|
throw new \Exception('Builder data by a empty string. need array');
|
||||||
|
|||||||
+19
-10
@@ -14,6 +14,7 @@ namespace Database;
|
|||||||
use Database\Affair\BeginTransaction;
|
use Database\Affair\BeginTransaction;
|
||||||
use Database\Affair\Commit;
|
use Database\Affair\Commit;
|
||||||
use Database\Affair\Rollback;
|
use Database\Affair\Rollback;
|
||||||
|
use Database\Base\Driver;
|
||||||
use Database\Base\PDO;
|
use Database\Base\PDO;
|
||||||
use Exception;
|
use Exception;
|
||||||
use Kiri\Abstracts\Component;
|
use Kiri\Abstracts\Component;
|
||||||
@@ -45,6 +46,7 @@ class Connection extends Component
|
|||||||
public string $charset = 'utf-8';
|
public string $charset = 'utf-8';
|
||||||
public string $tablePrefix = '';
|
public string $tablePrefix = '';
|
||||||
public string $database = '';
|
public string $database = '';
|
||||||
|
public string $driver = 'mysql';
|
||||||
public int $timeout = 30;
|
public int $timeout = 30;
|
||||||
public int $waite_time = 3;
|
public int $waite_time = 3;
|
||||||
public int $tick_time = 60000;
|
public int $tick_time = 60000;
|
||||||
@@ -154,7 +156,7 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
private function getName(): string
|
private function getName(): string
|
||||||
{
|
{
|
||||||
return 'mysql.' . $this->cds;
|
return strtolower($this->driver) . '.' . $this->cds;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -355,15 +357,22 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
public function newConnect(): \PDO
|
public function newConnect(): \PDO
|
||||||
{
|
{
|
||||||
$pdo = new PDO($this->database, $this->cds, $this->username, $this->password, [
|
$driver = strtolower($this->driver);
|
||||||
\PDO::ATTR_CASE => \PDO::CASE_NATURAL,
|
$options = [
|
||||||
\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,
|
||||||
]);
|
];
|
||||||
|
|
||||||
|
// MySQL 特定的选项
|
||||||
|
if ($driver === 'mysql') {
|
||||||
|
$options[\PDO::MYSQL_ATTR_INIT_COMMAND] = 'SET NAMES ' . $this->charset;
|
||||||
|
}
|
||||||
|
|
||||||
|
$pdo = new PDO($this->database, $this->cds, $this->username, $this->password, $options, $driver);
|
||||||
foreach ($this->attributes as $key => $attribute) {
|
foreach ($this->attributes as $key => $attribute) {
|
||||||
$pdo->setAttribute($key, $attribute);
|
$pdo->setAttribute($key, $attribute);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -78,6 +78,7 @@ class DatabasesProviders extends Providers
|
|||||||
'password' => $database['password'],
|
'password' => $database['password'],
|
||||||
'tablePrefix' => $database['tablePrefix'],
|
'tablePrefix' => $database['tablePrefix'],
|
||||||
'database' => $database['database'],
|
'database' => $database['database'],
|
||||||
|
'driver' => $database['driver'] ?? 'mysql',
|
||||||
'timeout' => $database['timeout'] ?? 10,
|
'timeout' => $database['timeout'] ?? 10,
|
||||||
'tick_time' => $database['tick_time'] ?? 60000,
|
'tick_time' => $database['tick_time'] ?? 60000,
|
||||||
'waite_time' => $database['waite_time'] ?? 3,
|
'waite_time' => $database['waite_time'] ?? 3,
|
||||||
|
|||||||
@@ -244,9 +244,26 @@ class SqlBuilder extends Component
|
|||||||
/**
|
/**
|
||||||
* @param string $table
|
* @param string $table
|
||||||
* @return string
|
* @return string
|
||||||
|
* @throws
|
||||||
*/
|
*/
|
||||||
public function columns(string $table): string
|
public function columns(string $table): string
|
||||||
{
|
{
|
||||||
|
$driver = $this->getDriver();
|
||||||
|
if (in_array($driver, ['pgsql', 'postgresql'])) {
|
||||||
|
// PostgreSQL 使用 information_schema
|
||||||
|
$tableName = trim($table, '"`');
|
||||||
|
return "SELECT
|
||||||
|
column_name as Field,
|
||||||
|
data_type as Type,
|
||||||
|
is_nullable as Null,
|
||||||
|
column_default as Default,
|
||||||
|
'' as Extra,
|
||||||
|
'' as Key
|
||||||
|
FROM information_schema.columns
|
||||||
|
WHERE table_name = '$tableName'
|
||||||
|
ORDER BY ordinal_position";
|
||||||
|
}
|
||||||
|
// MySQL 使用 SHOW FULL FIELDS
|
||||||
return 'SHOW FULL FIELDS FROM ' . $table;
|
return 'SHOW FULL FIELDS FROM ' . $table;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -317,9 +334,19 @@ class SqlBuilder extends Component
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
* @throws
|
||||||
|
*/
|
||||||
private function makeLimit(): string
|
private function makeLimit(): string
|
||||||
{
|
{
|
||||||
if ($this->query->getOffset() >= 0 && $this->query->getLimit() >= 1) {
|
if ($this->query->getOffset() >= 0 && $this->query->getLimit() >= 1) {
|
||||||
|
$driver = $this->getDriver();
|
||||||
|
if (in_array($driver, ['pgsql', 'postgresql'])) {
|
||||||
|
// PostgreSQL 使用 LIMIT ... OFFSET ... 语法
|
||||||
|
return ' LIMIT ' . $this->query->getLimit() . ' OFFSET ' . $this->query->getOffset();
|
||||||
|
}
|
||||||
|
// MySQL 使用 LIMIT offset,limit 语法
|
||||||
return ' LIMIT ' . $this->query->getOffset() . ',' . $this->query->getLimit();
|
return ' LIMIT ' . $this->query->getOffset() . ',' . $this->query->getLimit();
|
||||||
}
|
}
|
||||||
return '';
|
return '';
|
||||||
@@ -399,5 +426,36 @@ class SqlBuilder extends Component
|
|||||||
return $_array;
|
return $_array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取数据库驱动类型
|
||||||
|
* @return string
|
||||||
|
* @throws
|
||||||
|
*/
|
||||||
|
private function getDriver(): string
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
// 尝试从 query 对象获取 connection
|
||||||
|
if ($this->query instanceof ActiveQuery && $this->query->modelClass !== null) {
|
||||||
|
if ($this->query->modelClass instanceof Model) {
|
||||||
|
$connection = $this->query->modelClass->getConnection();
|
||||||
|
if ($connection instanceof Connection) {
|
||||||
|
return strtolower($connection->driver ?? 'mysql');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 如果是 Db 查询,尝试获取默认 connection
|
||||||
|
if (method_exists($this->query, 'getConnection')) {
|
||||||
|
$connection = $this->query->getConnection();
|
||||||
|
if ($connection instanceof Connection) {
|
||||||
|
return strtolower($connection->driver ?? 'mysql');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
// 忽略错误,返回默认值
|
||||||
|
}
|
||||||
|
// 默认返回 mysql
|
||||||
|
return 'mysql';
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,8 +12,6 @@ namespace Database\Traits;
|
|||||||
use Database\ModelInterface;
|
use Database\ModelInterface;
|
||||||
use Database\Collection;
|
use Database\Collection;
|
||||||
use Database\Relation;
|
use Database\Relation;
|
||||||
use Kiri;
|
|
||||||
use Kiri\Di\Context;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class HasBase
|
* Class HasBase
|
||||||
|
|||||||
Reference in New Issue
Block a user