|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This file is part of CodeIgniter 4 framework.
|
|
|
|
*
|
|
|
|
* (c) CodeIgniter Foundation <admin@codeigniter.com>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view
|
|
|
|
* the LICENSE file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace CodeIgniter\Database\Postgre;
|
|
|
|
|
|
|
|
use CodeIgniter\Database\BaseConnection;
|
|
|
|
use CodeIgniter\Database\Exceptions\DatabaseException;
|
|
|
|
use CodeIgniter\Database\RawSql;
|
|
|
|
use ErrorException;
|
|
|
|
use PgSql\Connection as PgSqlConnection;
|
|
|
|
use PgSql\Result as PgSqlResult;
|
|
|
|
use stdClass;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connection for Postgre
|
|
|
|
*
|
|
|
|
* @extends BaseConnection<PgSqlConnection, PgSqlResult>
|
|
|
|
*/
|
|
|
|
class Connection extends BaseConnection
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Database driver
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public $DBDriver = 'Postgre';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Database schema
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public $schema = 'public';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Identifier escape character
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public $escapeChar = '"';
|
|
|
|
|
|
|
|
protected $connect_timeout;
|
|
|
|
protected $options;
|
|
|
|
protected $sslmode;
|
|
|
|
protected $service;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connect to the database.
|
|
|
|
*
|
|
|
|
* @return false|resource
|
|
|
|
* @phpstan-return false|PgSqlConnection
|
|
|
|
*/
|
|
|
|
public function connect(bool $persistent = false)
|
|
|
|
{
|
|
|
|
if (empty($this->DSN)) {
|
|
|
|
$this->buildDSN();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert DSN string
|
|
|
|
// @TODO This format is for PDO_PGSQL.
|
|
|
|
// https://www.php.net/manual/en/ref.pdo-pgsql.connection.php
|
|
|
|
// Should deprecate?
|
|
|
|
if (mb_strpos($this->DSN, 'pgsql:') === 0) {
|
|
|
|
$this->convertDSN();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->connID = $persistent === true ? pg_pconnect($this->DSN) : pg_connect($this->DSN);
|
|
|
|
|
|
|
|
if ($this->connID !== false) {
|
|
|
|
if ($persistent === true && pg_connection_status($this->connID) === PGSQL_CONNECTION_BAD && pg_ping($this->connID) === false
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! empty($this->schema)) {
|
|
|
|
$this->simpleQuery("SET search_path TO {$this->schema},public");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->setClientEncoding($this->charset) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->connID;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts the DSN with semicolon syntax.
|
|
|
|
*/
|
|
|
|
private function convertDSN()
|
|
|
|
{
|
|
|
|
// Strip pgsql
|
|
|
|
$this->DSN = mb_substr($this->DSN, 6);
|
|
|
|
|
|
|
|
// Convert semicolons to spaces in DSN format like:
|
|
|
|
// pgsql:host=localhost;port=5432;dbname=database_name
|
|
|
|
// https://www.php.net/manual/en/function.pg-connect.php
|
|
|
|
$allowedParams = ['host', 'port', 'dbname', 'user', 'password', 'connect_timeout', 'options', 'sslmode', 'service'];
|
|
|
|
|
|
|
|
$parameters = explode(';', $this->DSN);
|
|
|
|
|
|
|
|
$output = '';
|
|
|
|
$previousParameter = '';
|
|
|
|
|
|
|
|
foreach ($parameters as $parameter) {
|
|
|
|
[$key, $value] = explode('=', $parameter, 2);
|
|
|
|
if (in_array($key, $allowedParams, true)) {
|
|
|
|
if ($previousParameter !== '') {
|
|
|
|
if (array_search($key, $allowedParams, true) < array_search($previousParameter, $allowedParams, true)) {
|
|
|
|
$output .= ';';
|
|
|
|
} else {
|
|
|
|
$output .= ' ';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$output .= $parameter;
|
|
|
|
$previousParameter = $key;
|
|
|
|
} else {
|
|
|
|
$output .= ';' . $parameter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->DSN = $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Keep or establish the connection if no queries have been sent for
|
|
|
|
* a length of time exceeding the server's idle timeout.
|
|
|
|
*/
|
|
|
|
public function reconnect()
|
|
|
|
{
|
|
|
|
if (pg_ping($this->connID) === false) {
|
|
|
|
$this->connID = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Close the database connection.
|
|
|
|
*/
|
|
|
|
protected function _close()
|
|
|
|
{
|
|
|
|
pg_close($this->connID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Select a specific database table to use.
|
|
|
|
*/
|
|
|
|
public function setDatabase(string $databaseName): bool
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a string containing the version of the database being used.
|
|
|
|
*/
|
|
|
|
public function getVersion(): string
|
|
|
|
{
|
|
|
|
if (isset($this->dataCache['version'])) {
|
|
|
|
return $this->dataCache['version'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! $this->connID) {
|
|
|
|
$this->initialize();
|
|
|
|
}
|
|
|
|
|
|
|
|
$pgVersion = pg_version($this->connID);
|
|
|
|
$this->dataCache['version'] = isset($pgVersion['server']) ?
|
|
|
|
(preg_match('/^(\d+\.\d+)/', $pgVersion['server'], $matches) ? $matches[1] : '') :
|
|
|
|
'';
|
|
|
|
|
|
|
|
return $this->dataCache['version'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Executes the query against the database.
|
|
|
|
*
|
|
|
|
* @return false|resource
|
|
|
|
* @phpstan-return false|PgSqlResult
|
|
|
|
*/
|
|
|
|
protected function execute(string $sql)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
return pg_query($this->connID, $sql);
|
|
|
|
} catch (ErrorException $e) {
|
|
|
|
log_message('error', (string) $e);
|
|
|
|
|
|
|
|
if ($this->DBDebug) {
|
|
|
|
throw new DatabaseException($e->getMessage(), $e->getCode(), $e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the prefix of the function to access the DB.
|
|
|
|
*/
|
|
|
|
protected function getDriverFunctionPrefix(): string
|
|
|
|
{
|
|
|
|
return 'pg_';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the total number of rows affected by this query.
|
|
|
|
*/
|
|
|
|
public function affectedRows(): int
|
|
|
|
{
|
|
|
|
return pg_affected_rows($this->resultID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "Smart" Escape String
|
|
|
|
*
|
|
|
|
* Escapes data based on type
|
|
|
|
*
|
|
|
|
* @param array|bool|float|int|object|string|null $str
|
|
|
|
*
|
|
|
|
* @return array|float|int|string
|
|
|
|
* @phpstan-return ($str is array ? array : float|int|string)
|
|
|
|
*/
|
|
|
|
public function escape($str)
|
|
|
|
{
|
|
|
|
if (! $this->connID) {
|
|
|
|
$this->initialize();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @psalm-suppress NoValue I don't know why ERROR. */
|
|
|
|
if (is_string($str) || (is_object($str) && method_exists($str, '__toString'))) {
|
|
|
|
if ($str instanceof RawSql) {
|
|
|
|
return $str->__toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
return pg_escape_literal($this->connID, $str);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_bool($str)) {
|
|
|
|
return $str ? 'TRUE' : 'FALSE';
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @psalm-suppress NoValue I don't know why ERROR. */
|
|
|
|
return parent::escape($str);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Platform-dependant string escape
|
|
|
|
*/
|
|
|
|
protected function _escapeString(string $str): string
|
|
|
|
{
|
|
|
|
if (! $this->connID) {
|
|
|
|
$this->initialize();
|
|
|
|
}
|
|
|
|
|
|
|
|
return pg_escape_string($this->connID, $str);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates the SQL for listing tables in a platform-dependent manner.
|
|
|
|
*
|
|
|
|
* @param string|null $tableName If $tableName is provided will return only this table if exists.
|
|
|
|
*/
|
|
|
|
protected function _listTables(bool $prefixLimit = false, ?string $tableName = null): string
|
|
|
|
{
|
|
|
|
$sql = 'SELECT "table_name" FROM "information_schema"."tables" WHERE "table_schema" = \'' . $this->schema . "'";
|
|
|
|
|
|
|
|
if ($tableName !== null) {
|
|
|
|
return $sql . ' AND "table_name" LIKE ' . $this->escape($tableName);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($prefixLimit !== false && $this->DBPrefix !== '') {
|
|
|
|
return $sql . ' AND "table_name" LIKE \''
|
|
|
|
. $this->escapeLikeString($this->DBPrefix) . "%' "
|
|
|
|
. sprintf($this->likeEscapeStr, $this->likeEscapeChar);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates a platform-specific query string so that the column names can be fetched.
|
|
|
|
*/
|
|
|
|
protected function _listColumns(string $table = ''): string
|
|
|
|
{
|
|
|
|
return 'SELECT "column_name"
|
|
|
|
FROM "information_schema"."columns"
|
|
|
|
WHERE LOWER("table_name") = '
|
|
|
|
. $this->escape($this->DBPrefix . strtolower($table))
|
|
|
|
. ' ORDER BY "ordinal_position"';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an array of objects with field data
|
|
|
|
*
|
|
|
|
* @return list<stdClass>
|
|
|
|
*
|
|
|
|
* @throws DatabaseException
|
|
|
|
*/
|
|
|
|
protected function _fieldData(string $table): array
|
|
|
|
{
|
|
|
|
$sql = 'SELECT "column_name", "data_type", "character_maximum_length", "numeric_precision", "column_default", "is_nullable"
|
|
|
|
FROM "information_schema"."columns"
|
|
|
|
WHERE LOWER("table_name") = '
|
|
|
|
. $this->escape(strtolower($table))
|
|
|
|
. ' ORDER BY "ordinal_position"';
|
|
|
|
|
|
|
|
if (($query = $this->query($sql)) === false) {
|
|
|
|
throw new DatabaseException(lang('Database.failGetFieldData'));
|
|
|
|
}
|
|
|
|
$query = $query->getResultObject();
|
|
|
|
|
|
|
|
$retVal = [];
|
|
|
|
|
|
|
|
for ($i = 0, $c = count($query); $i < $c; $i++) {
|
|
|
|
$retVal[$i] = new stdClass();
|
|
|
|
|
|
|
|
$retVal[$i]->name = $query[$i]->column_name;
|
|
|
|
$retVal[$i]->type = $query[$i]->data_type;
|
|
|
|
$retVal[$i]->max_length = $query[$i]->character_maximum_length > 0 ? $query[$i]->character_maximum_length : $query[$i]->numeric_precision;
|
|
|
|
$retVal[$i]->nullable = $query[$i]->is_nullable === 'YES';
|
|
|
|
$retVal[$i]->default = $query[$i]->column_default;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $retVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an array of objects with index data
|
|
|
|
*
|
|
|
|
* @return array<string, stdClass>
|
|
|
|
*
|
|
|
|
* @throws DatabaseException
|
|
|
|
*/
|
|
|
|
protected function _indexData(string $table): array
|
|
|
|
{
|
|
|
|
$sql = 'SELECT "indexname", "indexdef"
|
|
|
|
FROM "pg_indexes"
|
|
|
|
WHERE LOWER("tablename") = ' . $this->escape(strtolower($table)) . '
|
|
|
|
AND "schemaname" = ' . $this->escape('public');
|
|
|
|
|
|
|
|
if (($query = $this->query($sql)) === false) {
|
|
|
|
throw new DatabaseException(lang('Database.failGetIndexData'));
|
|
|
|
}
|
|
|
|
$query = $query->getResultObject();
|
|
|
|
|
|
|
|
$retVal = [];
|
|
|
|
|
|
|
|
foreach ($query as $row) {
|
|
|
|
$obj = new stdClass();
|
|
|
|
$obj->name = $row->indexname;
|
|
|
|
$_fields = explode(',', preg_replace('/^.*\((.+?)\)$/', '$1', trim($row->indexdef)));
|
|
|
|
$obj->fields = array_map(static fn ($v) => trim($v), $_fields);
|
|
|
|
|
|
|
|
if (strpos($row->indexdef, 'CREATE UNIQUE INDEX pk') === 0) {
|
|
|
|
$obj->type = 'PRIMARY';
|
|
|
|
} else {
|
|
|
|
$obj->type = (strpos($row->indexdef, 'CREATE UNIQUE') === 0) ? 'UNIQUE' : 'INDEX';
|
|
|
|
}
|
|
|
|
|
|
|
|
$retVal[$obj->name] = $obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $retVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an array of objects with Foreign key data
|
|
|
|
*
|
|
|
|
* @return array<string, stdClass>
|
|
|
|
*
|
|
|
|
* @throws DatabaseException
|
|
|
|
*/
|
|
|
|
protected function _foreignKeyData(string $table): array
|
|
|
|
{
|
|
|
|
$sql = 'SELECT c.constraint_name,
|
|
|
|
x.table_name,
|
|
|
|
x.column_name,
|
|
|
|
y.table_name as foreign_table_name,
|
|
|
|
y.column_name as foreign_column_name,
|
|
|
|
c.delete_rule,
|
|
|
|
c.update_rule,
|
|
|
|
c.match_option
|
|
|
|
FROM information_schema.referential_constraints c
|
|
|
|
JOIN information_schema.key_column_usage x
|
|
|
|
on x.constraint_name = c.constraint_name
|
|
|
|
JOIN information_schema.key_column_usage y
|
|
|
|
on y.ordinal_position = x.position_in_unique_constraint
|
|
|
|
and y.constraint_name = c.unique_constraint_name
|
|
|
|
WHERE x.table_name = ' . $this->escape($table) .
|
|
|
|
'order by c.constraint_name, x.ordinal_position';
|
|
|
|
|
|
|
|
if (($query = $this->query($sql)) === false) {
|
|
|
|
throw new DatabaseException(lang('Database.failGetForeignKeyData'));
|
|
|
|
}
|
|
|
|
|
|
|
|
$query = $query->getResultObject();
|
|
|
|
$indexes = [];
|
|
|
|
|
|
|
|
foreach ($query as $row) {
|
|
|
|
$indexes[$row->constraint_name]['constraint_name'] = $row->constraint_name;
|
|
|
|
$indexes[$row->constraint_name]['table_name'] = $table;
|
|
|
|
$indexes[$row->constraint_name]['column_name'][] = $row->column_name;
|
|
|
|
$indexes[$row->constraint_name]['foreign_table_name'] = $row->foreign_table_name;
|
|
|
|
$indexes[$row->constraint_name]['foreign_column_name'][] = $row->foreign_column_name;
|
|
|
|
$indexes[$row->constraint_name]['on_delete'] = $row->delete_rule;
|
|
|
|
$indexes[$row->constraint_name]['on_update'] = $row->update_rule;
|
|
|
|
$indexes[$row->constraint_name]['match'] = $row->match_option;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->foreignKeyDataToObjects($indexes);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns platform-specific SQL to disable foreign key checks.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function _disableForeignKeyChecks()
|
|
|
|
{
|
|
|
|
return 'SET CONSTRAINTS ALL DEFERRED';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns platform-specific SQL to enable foreign key checks.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function _enableForeignKeyChecks()
|
|
|
|
{
|
|
|
|
return 'SET CONSTRAINTS ALL IMMEDIATE;';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the last error code and message.
|
|
|
|
* Must return this format: ['code' => string|int, 'message' => string]
|
|
|
|
* intval(code) === 0 means "no error".
|
|
|
|
*
|
|
|
|
* @return array<string, int|string>
|
|
|
|
*/
|
|
|
|
public function error(): array
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'code' => '',
|
|
|
|
'message' => pg_last_error($this->connID) ?: '',
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return int|string
|
|
|
|
*/
|
|
|
|
public function insertID()
|
|
|
|
{
|
|
|
|
$v = pg_version($this->connID);
|
|
|
|
// 'server' key is only available since PostgreSQL 7.4
|
|
|
|
$v = explode(' ', $v['server'])[0] ?? 0;
|
|
|
|
|
|
|
|
$table = func_num_args() > 0 ? func_get_arg(0) : null;
|
|
|
|
$column = func_num_args() > 1 ? func_get_arg(1) : null;
|
|
|
|
|
|
|
|
if ($table === null && $v >= '8.1') {
|
|
|
|
$sql = 'SELECT LASTVAL() AS ins_id';
|
|
|
|
} elseif ($table !== null) {
|
|
|
|
if ($column !== null && $v >= '8.0') {
|
|
|
|
$sql = "SELECT pg_get_serial_sequence('{$table}', '{$column}') AS seq";
|
|
|
|
$query = $this->query($sql);
|
|
|
|
$query = $query->getRow();
|
|
|
|
$seq = $query->seq;
|
|
|
|
} else {
|
|
|
|
// seq_name passed in table parameter
|
|
|
|
$seq = $table;
|
|
|
|
}
|
|
|
|
|
|
|
|
$sql = "SELECT CURRVAL('{$seq}') AS ins_id";
|
|
|
|
} else {
|
|
|
|
return pg_last_oid($this->resultID);
|
|
|
|
}
|
|
|
|
|
|
|
|
$query = $this->query($sql);
|
|
|
|
$query = $query->getRow();
|
|
|
|
|
|
|
|
return (int) $query->ins_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build a DSN from the provided parameters
|
|
|
|
*/
|
|
|
|
protected function buildDSN()
|
|
|
|
{
|
|
|
|
if ($this->DSN !== '') {
|
|
|
|
$this->DSN = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
// If UNIX sockets are used, we shouldn't set a port
|
|
|
|
if (strpos($this->hostname, '/') !== false) {
|
|
|
|
$this->port = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->hostname !== '') {
|
|
|
|
$this->DSN = "host={$this->hostname} ";
|
|
|
|
}
|
|
|
|
|
|
|
|
// ctype_digit only accepts strings
|
|
|
|
$port = (string) $this->port;
|
|
|
|
|
|
|
|
if ($port !== '' && ctype_digit($port)) {
|
|
|
|
$this->DSN .= "port={$port} ";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->username !== '') {
|
|
|
|
$this->DSN .= "user={$this->username} ";
|
|
|
|
|
|
|
|
// An empty password is valid!
|
|
|
|
// password must be set to null to ignore it.
|
|
|
|
if ($this->password !== null) {
|
|
|
|
$this->DSN .= "password='{$this->password}' ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->database !== '') {
|
|
|
|
$this->DSN .= "dbname={$this->database} ";
|
|
|
|
}
|
|
|
|
|
|
|
|
// We don't have these options as elements in our standard configuration
|
|
|
|
// array, but they might be set by parse_url() if the configuration was
|
|
|
|
// provided via string> Example:
|
|
|
|
//
|
|
|
|
// Postgre://username:password@localhost:5432/database?connect_timeout=5&sslmode=1
|
|
|
|
foreach (['connect_timeout', 'options', 'sslmode', 'service'] as $key) {
|
|
|
|
if (isset($this->{$key}) && is_string($this->{$key}) && $this->{$key} !== '') {
|
|
|
|
$this->DSN .= "{$key}='{$this->{$key}}' ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->DSN = rtrim($this->DSN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set client encoding
|
|
|
|
*/
|
|
|
|
protected function setClientEncoding(string $charset): bool
|
|
|
|
{
|
|
|
|
return pg_set_client_encoding($this->connID, $charset) === 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Begin Transaction
|
|
|
|
*/
|
|
|
|
protected function _transBegin(): bool
|
|
|
|
{
|
|
|
|
return (bool) pg_query($this->connID, 'BEGIN');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Commit Transaction
|
|
|
|
*/
|
|
|
|
protected function _transCommit(): bool
|
|
|
|
{
|
|
|
|
return (bool) pg_query($this->connID, 'COMMIT');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Rollback Transaction
|
|
|
|
*/
|
|
|
|
protected function _transRollback(): bool
|
|
|
|
{
|
|
|
|
return (bool) pg_query($this->connID, 'ROLLBACK');
|
|
|
|
}
|
|
|
|
}
|