diff --git a/typo3/sysext/core/Classes/Database/SqlParser.php b/typo3/sysext/core/Classes/Database/SqlParser.php index a64df9ac4b244fc1f78b16d26fbd4569249dd9e3..7e391ae5dc7b7ef0504251275bbfdbd554f926d0 100644 --- a/typo3/sysext/core/Classes/Database/SqlParser.php +++ b/typo3/sysext/core/Classes/Database/SqlParser.php @@ -237,7 +237,7 @@ class SqlParser { // Get field/value pairs: while ($comma) { if ($fieldName = $this->nextPart($parseString, '^([[:alnum:]_]+)[[:space:]]*=')) { - // Strip of "=" sign. + // Strip off "=" sign. $this->nextPart($parseString, '^(=)'); $value = $this->getValue($parseString); $result['FIELDS'][$fieldName] = $value; @@ -739,7 +739,7 @@ class SqlParser { // Looking for a known function (only known functions supported) $func = $this->nextPart($parseString, '^(count|max|min|floor|sum|avg)[[:space:]]*\\('); if ($func) { - // Strip of "(" + // Strip off "(" $parseString = trim(substr($parseString, 1)); $stack[$pnt]['type'] = 'function'; $stack[$pnt]['function'] = $func; @@ -1037,7 +1037,7 @@ class SqlParser { // See if condition is EXISTS with a subquery if (preg_match('/^EXISTS[[:space:]]*[(]/i', $parseString)) { $stack[$level][$pnt[$level]]['func']['type'] = $this->nextPart($parseString, '^(EXISTS)[[:space:]]*'); - // Strip of "(" + // Strip off "(" $parseString = trim(substr($parseString, 1)); $stack[$level][$pnt[$level]]['func']['subquery'] = $this->parseSELECT($parseString, $parameterReferences); // Seek to new position in parseString after parsing of the subquery @@ -1050,7 +1050,7 @@ class SqlParser { // See if LOCATE function is found if (preg_match('/^LOCATE[[:space:]]*[(]/i', $parseString)) { $stack[$level][$pnt[$level]]['func']['type'] = $this->nextPart($parseString, '^(LOCATE)[[:space:]]*'); - // Strip of "(" + // Strip off "(" $parseString = trim(substr($parseString, 1)); $stack[$level][$pnt[$level]]['func']['substr'] = $this->getValue($parseString); if (!$this->nextPart($parseString, '^(,)')) { @@ -1078,7 +1078,7 @@ class SqlParser { } elseif (preg_match('/^IFNULL[[:space:]]*[(]/i', $parseString)) { $stack[$level][$pnt[$level]]['func']['type'] = $this->nextPart($parseString, '^(IFNULL)[[:space:]]*'); $parseString = trim(substr($parseString, 1)); - // Strip of "(" + // Strip off "(" if ($fieldName = $this->nextPart($parseString, '^([[:alnum:]\\*._]+)[[:space:]]*')) { // Parse field name into field and table: $tableField = explode('.', $fieldName, 2); @@ -1098,9 +1098,32 @@ class SqlParser { if (!$this->nextPart($parseString, '^([)])')) { return $this->parseError('No ) parenthesis at end of function', $parseString); } + } elseif (preg_match('/^CAST[[:space:]]*[(]/i', $parseString)) { + $stack[$level][$pnt[$level]]['func']['type'] = $this->nextPart($parseString, '^(CAST)[[:space:]]*'); + $parseString = trim(substr($parseString, 1)); + // Strip off "(" + if ($fieldName = $this->nextPart($parseString, '^([[:alnum:]\\*._]+)[[:space:]]*')) { + // Parse field name into field and table: + $tableField = explode('.', $fieldName, 2); + if (count($tableField) === 2) { + $stack[$level][$pnt[$level]]['func']['table'] = $tableField[0]; + $stack[$level][$pnt[$level]]['func']['field'] = $tableField[1]; + } else { + $stack[$level][$pnt[$level]]['func']['table'] = ''; + $stack[$level][$pnt[$level]]['func']['field'] = $tableField[0]; + } + } else { + return $this->parseError('No field name found as expected in parseWhereClause()', $parseString); + } + if ($this->nextPart($parseString, '^([[:space:]]*AS[[:space:]]*)')) { + $stack[$level][$pnt[$level]]['func']['datatype'] = $this->getValue($parseString); + } + if (!$this->nextPart($parseString, '^([)])')) { + return $this->parseError('No ) parenthesis at end of function', $parseString); + } } elseif (preg_match('/^FIND_IN_SET[[:space:]]*[(]/i', $parseString)) { $stack[$level][$pnt[$level]]['func']['type'] = $this->nextPart($parseString, '^(FIND_IN_SET)[[:space:]]*'); - // Strip of "(" + // Strip off "(" $parseString = trim(substr($parseString, 1)); if ($str = $this->getValue($parseString)) { $stack[$level][$pnt[$level]]['func']['str'] = $str; @@ -1917,6 +1940,11 @@ class SqlParser { $output .= ($v['func']['table'] ? $v['func']['table'] . '.' : '') . $v['func']['field']; $output .= ', ' . $v['func']['default'][1] . $this->compileAddslashes($v['func']['default'][0]) . $v['func']['default'][1]; $output .= ')'; + } elseif (isset($v['func']) && $v['func']['type'] === 'CAST') { + $output .= ' ' . trim($v['modifier']) . ' CAST('; + $output .= ($v['func']['table'] ? $v['func']['table'] . '.' : '') . $v['func']['field']; + $output .= ' AS ' . $v['func']['datatype'][0]; + $output .= ')'; } elseif (isset($v['func']) && $v['func']['type'] === 'FIND_IN_SET') { $output .= ' ' . trim($v['modifier']) . ' FIND_IN_SET('; $output .= $v['func']['str'][1] . $v['func']['str'][0] . $v['func']['str'][1]; diff --git a/typo3/sysext/core/Classes/Tree/TableConfiguration/DatabaseTreeDataProvider.php b/typo3/sysext/core/Classes/Tree/TableConfiguration/DatabaseTreeDataProvider.php index de929b490833dc6f0e3a68c001d5c9f8db98edd9..3241c62e991e6186781c572b3e18e0f0c0772be3 100644 --- a/typo3/sysext/core/Classes/Tree/TableConfiguration/DatabaseTreeDataProvider.php +++ b/typo3/sysext/core/Classes/Tree/TableConfiguration/DatabaseTreeDataProvider.php @@ -428,7 +428,7 @@ class DatabaseTreeDataProvider extends AbstractTableConfigurationTreeDataProvide * @return int[] all uids found */ protected function listFieldQuery($fieldName, $queryId) { - $records = $GLOBALS['TYPO3_DB']->exec_SELECTgetRows('uid', $this->getTableName(), $GLOBALS['TYPO3_DB']->listQuery($fieldName, (int)$queryId, $this->getTableName()) . ((int)$queryId == 0 ? ' OR ' . $fieldName . ' = \'\'' : '')); + $records = $GLOBALS['TYPO3_DB']->exec_SELECTgetRows('uid', $this->getTableName(), $GLOBALS['TYPO3_DB']->listQuery($fieldName, (int)$queryId, $this->getTableName()) . ((int)$queryId === 0 ? ' OR CAST(' . $fieldName . ' AS CHAR) = \'\'' : '')); $uidArray = array(); foreach ($records as $record) { $uidArray[] = $record['uid']; diff --git a/typo3/sysext/core/Tests/Unit/Database/SqlParserTest.php b/typo3/sysext/core/Tests/Unit/Database/SqlParserTest.php index 990afc99f32c0e12f129dc5e95ddef30c45f67c3..115287d91549c1c6b99501d53594a49a1f01d90b 100644 --- a/typo3/sysext/core/Tests/Unit/Database/SqlParserTest.php +++ b/typo3/sysext/core/Tests/Unit/Database/SqlParserTest.php @@ -98,6 +98,21 @@ class SqlParserTest extends \TYPO3\CMS\Core\Tests\UnitTestCase { 'field' => 'fe_group' ), 'comparator' => '' + ), + 5 => array( + 'operator' => 'OR', + 'modifier' => '', + 'func' => array( + 'type' => 'CAST', + 'table' => 'pages', + 'field' => 'fe_group', + 'datatype' => 'CHAR' + ), + 'comparator' => '=', + 'value' => array( + 0 => '', + 1 => '\'' + ) ) ); $output = $this->subject->compileWhereClause($clauses); diff --git a/typo3/sysext/dbal/Classes/Database/DatabaseConnection.php b/typo3/sysext/dbal/Classes/Database/DatabaseConnection.php index a51661c8fda62ebb6ded6c7f2f0349b402584d80..8d53f120456a2f0a296efcda215960e3f6d27b87 100644 --- a/typo3/sysext/dbal/Classes/Database/DatabaseConnection.php +++ b/typo3/sysext/dbal/Classes/Database/DatabaseConnection.php @@ -1753,7 +1753,17 @@ class DatabaseConnection extends \TYPO3\CMS\Core\Database\DatabaseConnection { // but it's not overridden from \TYPO3\CMS\Core\Database\DatabaseConnection at the moment... $patternForLike = $this->escapeStrForLike($pattern, $where_clause[$k]['func']['table']); $where_clause[$k]['func']['str_like'] = $patternForLike; - // Intentional fallthrough + if ($where_clause[$k]['func']['table'] !== '') { + $where_clause[$k]['func']['table'] = $this->quoteName($v['func']['table']); + } + if ($where_clause[$k]['func']['field'] !== '') { + if ($this->dbmsSpecifics->getSpecific(Specifics\AbstractSpecifics::CAST_FIND_IN_SET)) { + $where_clause[$k]['func']['field'] = 'CAST(' . $this->quoteName($v['func']['field']) . ' AS CHAR)'; + } else { + $where_clause[$k]['func']['field'] = $this->quoteName($v['func']['field']); + } + } + break; case 'IFNULL': // Intentional fallthrough case 'LOCATE': diff --git a/typo3/sysext/dbal/Classes/Database/Specifics/AbstractSpecifics.php b/typo3/sysext/dbal/Classes/Database/Specifics/AbstractSpecifics.php index 722aaeb4511116a04bbd3d2aa777421959c84eda..b2e2f947d6af0d3818f1a57fee71463d4357eb0b 100644 --- a/typo3/sysext/dbal/Classes/Database/Specifics/AbstractSpecifics.php +++ b/typo3/sysext/dbal/Classes/Database/Specifics/AbstractSpecifics.php @@ -26,6 +26,7 @@ abstract class AbstractSpecifics { const FIELD_MAXLENGTH = 'field_maxlength'; const LIST_MAXEXPRESSIONS = 'list_maxexpressions'; const PARTIAL_STRING_INDEX = 'partial_string_index'; + const CAST_FIND_IN_SET = 'cast_find_in_set'; /** * Contains the specifics of a DBMS. diff --git a/typo3/sysext/dbal/Classes/Database/Specifics/PostgresSpecifics.php b/typo3/sysext/dbal/Classes/Database/Specifics/PostgresSpecifics.php index 6f9138f28b2480d59d51b6478b64bb37e6efb720..145c7ef2bb9492eef96e6909eca76fc524f21731 100644 --- a/typo3/sysext/dbal/Classes/Database/Specifics/PostgresSpecifics.php +++ b/typo3/sysext/dbal/Classes/Database/Specifics/PostgresSpecifics.php @@ -21,6 +21,15 @@ use TYPO3\CMS\Core\Utility\GeneralUtility; * Any logic is in AbstractSpecifics. */ class PostgresSpecifics extends AbstractSpecifics { + /** + * Contains the specifics that need to be taken care of for PostgreSQL DBMS. + * + * @var array + */ + protected $specificProperties = array( + self::CAST_FIND_IN_SET => TRUE + ); + /** * Contains the DBMS specific mapping overrides for native MySQL to ADOdb meta field types */ diff --git a/typo3/sysext/dbal/Tests/Unit/Database/DatabaseConnectionPostgresqlTest.php b/typo3/sysext/dbal/Tests/Unit/Database/DatabaseConnectionPostgresqlTest.php index 046eaf634ad6f2b04edc924d5d486ffa246ae693..638f1f306f81a28d0ab88358841867d30ccd0818 100644 --- a/typo3/sysext/dbal/Tests/Unit/Database/DatabaseConnectionPostgresqlTest.php +++ b/typo3/sysext/dbal/Tests/Unit/Database/DatabaseConnectionPostgresqlTest.php @@ -90,7 +90,7 @@ class DatabaseConnectionPostgresqlTest extends AbstractTestCase { */ public function findInSetIsProperlyRemapped() { $result = $this->subject->SELECTquery('*', 'fe_users', 'FIND_IN_SET(10, usergroup)'); - $expected = 'SELECT * FROM "fe_users" WHERE FIND_IN_SET(10, "usergroup") != 0'; + $expected = 'SELECT * FROM "fe_users" WHERE FIND_IN_SET(10, CAST("usergroup" AS CHAR)) != 0'; $this->assertEquals($expected, $this->cleanSql($result)); } diff --git a/typo3/sysext/dbal/Tests/Unit/Database/SqlParserTest.php b/typo3/sysext/dbal/Tests/Unit/Database/SqlParserTest.php index 9e1a948bf1511613d6edd497e391ef6b8e22542f..f5b0cd7c3414f7057c08f59986a32bcaac64e0f2 100644 --- a/typo3/sysext/dbal/Tests/Unit/Database/SqlParserTest.php +++ b/typo3/sysext/dbal/Tests/Unit/Database/SqlParserTest.php @@ -269,6 +269,31 @@ class SqlParserTest extends AbstractTestCase { $this->assertEquals($expected, $this->cleanSql($result)); } + /** + * @test + * @see http://forge.typo3.org/issues/67155 + */ + public function canParseCastOperator() { + $parseString = 'CAST(parent AS CHAR) != \'\''; + $result = $this->subject->parseWhereClause($parseString); + $this->assertInternalType('array', $result); + $this->assertEmpty($parseString); + } + + /** + * @test + * @see http://forge.typo3.org/issues/67155 + */ + public function canCompileCastOperator() { + $parseString = 'SELECT * FROM sys_category WHERE CAST(parent AS CHAR) != \'\''; + $components = $this->subject->_callRef('parseSELECT', $parseString); + $this->assertInternalType('array', $components); + + $result = $this->subject->_callRef('compileSELECT', $components); + $expected = 'SELECT * FROM sys_category WHERE CAST(parent AS CHAR) != \'\''; + $this->assertEquals($expected, $this->cleanSql($result)); + } + /** * @test * @see http://forge.typo3.org/issues/22695