* Api/MariaDB/MySQL: fix SQL error handling in PHP 8.1

also introduce parameter to limit number of deleted rows and more docu
This commit is contained in:
ralf 2023-03-17 11:04:45 +01:00
parent 69ca5f4f31
commit a0a1ccf0e1

View File

@ -477,8 +477,6 @@ class Db
break;
case 'sapdb':
$Type = 'maxdb';
// fall through
case 'maxdb':
$Type ='sapdb'; // name in ADOdb
$php_extension = 'odbc';
@ -750,7 +748,8 @@ class Db
* @param int $fetchmode =self::FETCH_BOTH self::FETCH_BOTH (default), self::FETCH_ASSOC or self::FETCH_NUM
* @param boolean $reconnect =true true: try reconnecting if server closes connection, false: dont (mysql only!)
* @return ADORecordSet or false, if the query fails
* @throws Db\Exception\InvalidSql with $this->Link_ID->ErrorNo() as code
* @throws Db\Exception\InvalidSql for SQL syntax errors
* @throws Db\Exception with $this->Link_ID->ErrorNo() as code for all other errors
*/
function query($Query_String, $line = '', $file = '', $offset=0, $num_rows=-1, $inputarr=false, $fetchmode=self::FETCH_BOTH, $reconnect=true)
{
@ -797,9 +796,16 @@ class Db
// PHP 8.1 mysqli throws its own exception
catch(\mysqli_sql_exception $e) {
if (!($reconnect && $this->Type == 'mysql' && ($e->getCode() == 2006 || $e->getMessage() === 'MySQL server has gone away')))
{
if ($e->getCode() == 1064) // You have an error in your SQL syntax
{
throw new Db\Exception\InvalidSql($e->getMessage(), $e->getCode(), $e);
}
else
{
throw new Db\Exception($e->getMessage(), $e->getCode(), $e);
}
}
$this->Errno = 2006;
$this->Error = $e->getMessage();
}
@ -826,7 +832,7 @@ class Db
"\n$this->Error ($this->Errno)".
($inputarr ? "\nParameters: '".implode("','",$inputarr)."'":''), $this->Errno);
}
elseif(empty($rs->sql)) $rs->sql = $Query_String;
elseif(!$rs->sql) $rs->sql = $Query_String;
return $rs;
}
@ -1799,7 +1805,7 @@ class Db
if ($app === true && $table)
{
foreach(self::$all_app_data as $app => &$app_data)
foreach(self::$all_app_data as &$app_data)
{
if (isset($app_data[$table]))
{
@ -1807,27 +1813,27 @@ class Db
}
}
// $table not found in loaded apps, check not yet loaded ones
foreach(scandir(EGW_INCLUDE_ROOT) as $app)
foreach(scandir(EGW_INCLUDE_ROOT) as $dir)
{
if ($app[0] == '.' || !is_dir(EGW_INCLUDE_ROOT.'/'.$app) || isset(self::$all_app_data[$app]))
if ($dir[0] == '.' || !is_dir(EGW_INCLUDE_ROOT.'/'.$dir) || isset(self::$all_app_data[$dir]))
{
continue;
}
$tables_current = EGW_INCLUDE_ROOT . "/$app/setup/tables_current.inc.php";
$tables_current = EGW_INCLUDE_ROOT . "/$dir/setup/tables_current.inc.php";
if (!@file_exists($tables_current))
{
self::$all_app_data[$app] = False;
self::$all_app_data[$dir] = False;
}
else
{
$phpgw_baseline = null;
include($tables_current);
self::$all_app_data[$app] =& $phpgw_baseline;
self::$all_app_data[$dir] =& $phpgw_baseline;
unset($phpgw_baseline);
if (isset(self::$all_app_data[$app][$table]))
if (isset(self::$all_app_data[$dir][$table]))
{
return self::$all_app_data[$app][$table];
return self::$all_app_data[$dir][$table];
}
}
}
@ -1835,7 +1841,7 @@ class Db
}
if (!$app)
{
$app = $this->app ? $this->app : $GLOBALS['egw_info']['flags']['currentapp'];
$app = $this->app ?: $GLOBALS['egw_info']['flags']['currentapp'];
}
$app_data =& self::$all_app_data[$app];
@ -1899,6 +1905,9 @@ class Db
* @param string|boolean $app string with name of app or False to use the current-app
* @param bool $use_prepared_statement use a prepared statement
* @param array|bool $table_def use this table definition. If False, the table definition will be read from tables_baseline
* @throws Db\Exception\InvalidSql for SQL syntax errors
* @throws Db\Exception with $this->Link_ID->ErrorNo() as code for all other errors
* @throws Exception\WrongParameter use $where together with multiple data rows in $data
* @return ADORecordSet or false, if the query fails
*/
function insert($table,$data,$where,$line,$file,$app=False,$use_prepared_statement=false,$table_def=False)
@ -2008,6 +2017,8 @@ class Db
* @param string|boolean $app string with name of app or False to use the current-app
* @param bool $use_prepared_statement use a prepared statement
* @param array|bool $table_def use this table definition. If False, the table definition will be read from tables_baseline
* @throws Db\Exception\InvalidSql for SQL syntax errors
* @throws Db\Exception with $this->Link_ID->ErrorNo() as code for all other errors
* @return ADORecordSet or false, if the query fails
*/
function update($table,$data,$where,$line,$file,$app=False,$use_prepared_statement=false,$table_def=False)
@ -2096,9 +2107,12 @@ class Db
* @param string $file file-name to pass to query
* @param string|boolean $app string with name of app or False to use the current-app
* @param array|bool $table_def use this table definition. If False, the table definition will be read from tables_baseline
* @param int|null $limit limit delete to given number of rows
* @throws Db\Exception\InvalidSql for SQL syntax errors
* @throws Db\Exception e.g. 1205: Lock timeout exceeded, if deleting rows took to long, you can use $limit to delete in multiple calls
* @return ADORecordSet or false, if the query fails
*/
function delete($table,$where,$line,$file,$app=False,$table_def=False)
function delete($table, $where, $line, $file, $app=False, $table_def=False, int $limit=null)
{
if (!$table_def) $table_def = $this->get_table_definitions($app,$table);
@ -2107,7 +2121,8 @@ class Db
$table = self::$tablealiases[$table];
}
$sql = "DELETE FROM $table WHERE ".
$this->column_data_implode(' AND ',$where,True,False,$table_def['fd']);
$this->column_data_implode(' AND ',$where,True,False,$table_def['fd']).
(isset($limit) && $limit > 0 ? 'LIMIT '.$limit : '');
return $this->query($sql,$line,$file);
}
@ -2180,6 +2195,8 @@ class Db
* "LEFT JOIN table2 ON (x=y)", Note: there's no quoting done on $join!
* @param array|bool $table_def use this table definition. If False, the table definition will be read from tables_baseline
* @param int $fetchmode =self::FETCH_ASSOC self::FETCH_ASSOC (default), self::FETCH_BOTH or self::FETCH_NUM
* @throws Db\Exception\InvalidSql for SQL syntax errors
* @throws Db\Exception with $this->Link_ID->ErrorNo() as code for all other errors
* @return ADORecordSet or false, if the query fails
*/
function select($table,$cols,$where,$line,$file,$offset=False,$append='',$app=False,$num_rows=0,$join='',$table_def=False,$fetchmode=self::FETCH_ASSOC)
@ -2228,6 +2245,8 @@ class Db
* @param int|bool $offset offset for a limited query or False (default)
* @param int $num_rows number of rows to return if offset set, default 0 = use default in user prefs
* @param int $fetchmode =self::FETCH_ASSOC self::FETCH_ASSOC (default), self::FETCH_BOTH or self::FETCH_NUM
* @throws Db\Exception\InvalidSql for SQL syntax errors
* @throws Db\Exception with $this->Link_ID->ErrorNo() as code for all other errors
* @return ADORecordSet or false, if the query fails
*/
function union($selects,$line,$file,$order_by='',$offset=false,$num_rows=0,$fetchmode=self::FETCH_ASSOC)