1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_dbaccess.hxx" 26 27 #include "composertools.hxx" 28 #include "core_resource.hrc" 29 #include "core_resource.hxx" 30 #include "dbastrings.hrc" 31 #include "HelperCollections.hxx" 32 #include "SingleSelectQueryComposer.hxx" 33 #include "sdbcoretools.hxx" 34 35 /** === begin UNO includes === **/ 36 #include <com/sun/star/beans/PropertyAttribute.hpp> 37 #include <com/sun/star/container/XChild.hpp> 38 #include <com/sun/star/i18n/XLocaleData.hpp> 39 #include <com/sun/star/lang/DisposedException.hpp> 40 #include <com/sun/star/sdb/BooleanComparisonMode.hpp> 41 #include <com/sun/star/sdb/SQLFilterOperator.hpp> 42 #include <com/sun/star/sdb/XQueriesSupplier.hpp> 43 #include <com/sun/star/sdb/CommandType.hpp> 44 #include <com/sun/star/sdbc/ColumnSearch.hpp> 45 #include <com/sun/star/sdbc/DataType.hpp> 46 #include <com/sun/star/sdbc/XResultSetMetaData.hpp> 47 #include <com/sun/star/sdbc/XResultSetMetaDataSupplier.hpp> 48 #include <com/sun/star/sdbc/XParameters.hpp> 49 #include <com/sun/star/uno/XAggregation.hpp> 50 #include <com/sun/star/util/XNumberFormatter.hpp> 51 /** === end UNO includes === **/ 52 53 #include <comphelper/processfactory.hxx> 54 #include <comphelper/sequence.hxx> 55 #include <comphelper/types.hxx> 56 #include <cppuhelper/typeprovider.hxx> 57 #include <connectivity/predicateinput.hxx> 58 #include <rtl/logfile.hxx> 59 #include <unotools/syslocale.hxx> 60 #include <tools/debug.hxx> 61 #include <tools/diagnose_ex.h> 62 #include <unotools/configmgr.hxx> 63 #include <unotools/sharedunocomponent.hxx> 64 65 #include <memory> 66 67 using namespace ::dbaccess; 68 using namespace ::dbtools; 69 using namespace ::comphelper; 70 using namespace ::connectivity; 71 using namespace ::com::sun::star::uno; 72 using namespace ::com::sun::star::beans; 73 using namespace ::com::sun::star::sdbc; 74 using namespace ::com::sun::star::sdb; 75 using namespace ::com::sun::star::sdbcx; 76 using namespace ::com::sun::star::container; 77 using namespace ::com::sun::star::i18n; 78 using namespace ::com::sun::star::lang; 79 using namespace ::com::sun::star::script; 80 using namespace ::com::sun::star::util; 81 using namespace ::cppu; 82 using namespace ::osl; 83 using namespace ::utl; 84 85 namespace dbaccess { 86 namespace BooleanComparisonMode = ::com::sun::star::sdb::BooleanComparisonMode; 87 } 88 89 #define STR_SELECT ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("SELECT ")) 90 #define STR_FROM ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" FROM ")) 91 #define STR_WHERE ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" WHERE ")) 92 #define STR_GROUP_BY ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" GROUP BY ")) 93 #define STR_HAVING ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" HAVING ")) 94 #define STR_ORDER_BY ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" ORDER BY ")) 95 #define STR_AND ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" AND ")) 96 #define STR_OR ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" OR ")) 97 #define STR_LIKE ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" LIKE ")) 98 #define STR_EQUAL ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" = ")) 99 #define L_BRACKET ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("(")) 100 #define R_BRACKET ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(")")) 101 #define COMMA ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(",")) 102 103 // ------------------------------------------------------------------------- 104 namespace 105 { 106 // ..................................................................... 107 /** parses the given statement, using the given parser, returns a parse node representing 108 the statement 109 110 If the statement cannot be parsed, an error is thrown. 111 */ 112 const OSQLParseNode* parseStatement_throwError( OSQLParser& _rParser, const ::rtl::OUString& _rStatement, const Reference< XInterface >& _rxContext ) 113 { 114 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "frank.schoenheit@sun.com", "SingleSelectQueryComposer.cxx::parseStatement_throwError" ); 115 ::rtl::OUString aErrorMsg; 116 const OSQLParseNode* pNewSqlParseNode = _rParser.parseTree( aErrorMsg, _rStatement ); 117 if ( !pNewSqlParseNode ) 118 { 119 ::rtl::OUString sSQLStateGeneralError( getStandardSQLState( SQL_GENERAL_ERROR ) ); 120 SQLException aError2( aErrorMsg, _rxContext, sSQLStateGeneralError, 1000, Any() ); 121 SQLException aError1( _rStatement, _rxContext, sSQLStateGeneralError, 1000, makeAny( aError2 ) ); 122 throw SQLException(_rParser.getContext().getErrorMessage(OParseContext::ERROR_GENERAL),_rxContext,sSQLStateGeneralError,1000,makeAny(aError1)); 123 } 124 return pNewSqlParseNode; 125 } 126 127 // ..................................................................... 128 /** checks whether the given parse node describes a valid single select statement, throws 129 an error if not 130 */ 131 void checkForSingleSelect_throwError( const OSQLParseNode* pStatementNode, OSQLParseTreeIterator& _rIterator, 132 const Reference< XInterface >& _rxContext, const ::rtl::OUString& _rOriginatingCommand ) 133 { 134 const OSQLParseNode* pOldNode = _rIterator.getParseTree(); 135 136 // determine the statement type 137 _rIterator.setParseTree( pStatementNode ); 138 _rIterator.traverseAll(); 139 bool bIsSingleSelect = ( _rIterator.getStatementType() == SQL_STATEMENT_SELECT ); 140 141 // throw the error, if necessary 142 if ( !bIsSingleSelect || SQL_ISRULE( pStatementNode, union_statement ) ) // #i4229# OJ 143 { 144 // restore the old node before throwing the exception 145 _rIterator.setParseTree( pOldNode ); 146 // and now really ... 147 SQLException aError1( _rOriginatingCommand, _rxContext, getStandardSQLState( SQL_GENERAL_ERROR ), 1000, Any() ); 148 throw SQLException( DBACORE_RESSTRING( RID_STR_ONLY_QUERY ), _rxContext, 149 getStandardSQLState( SQL_GENERAL_ERROR ), 1000, makeAny( aError1 ) ); 150 } 151 152 delete pOldNode; 153 } 154 155 // ..................................................................... 156 /** combines parseStatement_throwError and checkForSingleSelect_throwError 157 */ 158 void parseAndCheck_throwError( OSQLParser& _rParser, const ::rtl::OUString& _rStatement, 159 OSQLParseTreeIterator& _rIterator, const Reference< XInterface >& _rxContext ) 160 { 161 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "frank.schoenheit@sun.com", "SingleSelectQueryComposer.cxx::parseAndCheck_throwError" ); 162 const OSQLParseNode* pNode = parseStatement_throwError( _rParser, _rStatement, _rxContext ); 163 checkForSingleSelect_throwError( pNode, _rIterator, _rxContext, _rStatement ); 164 } 165 166 // ..................................................................... 167 /** transforms a parse node describing a complete statement into a pure select 168 statement, without any filter/order/groupby/having clauses 169 */ 170 ::rtl::OUString getPureSelectStatement( const OSQLParseNode* _pRootNode, Reference< XConnection > _rxConnection ) 171 { 172 ::rtl::OUString sSQL = STR_SELECT; 173 _pRootNode->getChild(1)->parseNodeToStr( sSQL, _rxConnection ); 174 _pRootNode->getChild(2)->parseNodeToStr( sSQL, _rxConnection ); 175 sSQL += STR_FROM; 176 _pRootNode->getChild(3)->getChild(0)->getChild(1)->parseNodeToStr( sSQL, _rxConnection ); 177 return sSQL; 178 } 179 180 /** resets an SQL iterator, including deletion of the parse tree, and disposal if desired 181 */ 182 void resetIterator( OSQLParseTreeIterator& _rIterator, bool _bDispose ) 183 { 184 const OSQLParseNode* pSqlParseNode = _rIterator.getParseTree(); 185 _rIterator.setParseTree(NULL); 186 delete pSqlParseNode; 187 if ( _bDispose ) 188 _rIterator.dispose(); 189 } 190 void lcl_addFilterCriteria_throw(sal_Int32 i_nFilterOperator,const ::rtl::OUString& i_sValue,::rtl::OUStringBuffer& o_sRet) 191 { 192 switch( i_nFilterOperator ) 193 { 194 case SQLFilterOperator::EQUAL: 195 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" = "))); 196 o_sRet.append(i_sValue); 197 break; 198 case SQLFilterOperator::NOT_EQUAL: 199 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" <> "))); 200 o_sRet.append(i_sValue); 201 break; 202 case SQLFilterOperator::LESS: 203 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" < "))); 204 o_sRet.append(i_sValue); 205 break; 206 case SQLFilterOperator::GREATER: 207 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" > "))); 208 o_sRet.append(i_sValue); 209 break; 210 case SQLFilterOperator::LESS_EQUAL: 211 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" <= "))); 212 o_sRet.append(i_sValue); 213 break; 214 case SQLFilterOperator::GREATER_EQUAL: 215 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" >= "))); 216 o_sRet.append(i_sValue); 217 break; 218 case SQLFilterOperator::LIKE: 219 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" LIKE "))); 220 o_sRet.append(i_sValue); 221 break; 222 case SQLFilterOperator::NOT_LIKE: 223 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" NOT LIKE "))); 224 o_sRet.append(i_sValue); 225 break; 226 case SQLFilterOperator::SQLNULL: 227 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" IS NULL")) ); 228 break; 229 case SQLFilterOperator::NOT_SQLNULL: 230 o_sRet.append(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" IS NOT NULL")) ); 231 break; 232 default: 233 throw SQLException(); 234 } 235 } 236 237 } 238 239 DBG_NAME(OSingleSelectQueryComposer) 240 // ------------------------------------------------------------------------- 241 OSingleSelectQueryComposer::OSingleSelectQueryComposer(const Reference< XNameAccess>& _rxTables, 242 const Reference< XConnection>& _xConnection, 243 const ::comphelper::ComponentContext& _rContext ) 244 :OSubComponent(m_aMutex,_xConnection) 245 ,OPropertyContainer(m_aBHelper) 246 ,m_aSqlParser( _rContext.getLegacyServiceFactory() ) 247 ,m_aSqlIterator( _xConnection, _rxTables, m_aSqlParser, NULL ) 248 ,m_aAdditiveIterator( _xConnection, _rxTables, m_aSqlParser, NULL ) 249 ,m_aElementaryParts( (size_t)SQLPartCount ) 250 ,m_xConnection(_xConnection) 251 ,m_xMetaData(_xConnection->getMetaData()) 252 ,m_xConnectionTables( _rxTables ) 253 ,m_aContext( _rContext ) 254 ,m_pTables(NULL) 255 ,m_nBoolCompareMode( BooleanComparisonMode::EQUAL_INTEGER ) 256 ,m_nCommandType(CommandType::COMMAND) 257 { 258 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::OSingleSelectQueryComposer" ); 259 DBG_CTOR(OSingleSelectQueryComposer,NULL); 260 261 if ( !m_aContext.is() || !m_xConnection.is() || !m_xConnectionTables.is() ) 262 throw IllegalArgumentException(); 263 264 registerProperty(PROPERTY_ORIGINAL,PROPERTY_ID_ORIGINAL,PropertyAttribute::BOUND|PropertyAttribute::READONLY,&m_sOrignal,::getCppuType(&m_sOrignal)); 265 266 m_aCurrentColumns.resize(4); 267 268 m_aLocale = SvtSysLocale().GetLocaleData().getLocale(); 269 m_xNumberFormatsSupplier = dbtools::getNumberFormats( m_xConnection, sal_True, m_aContext.getLegacyServiceFactory() ); 270 Reference< XLocaleData > xLocaleData; 271 m_aContext.createComponent( "com.sun.star.i18n.LocaleData", xLocaleData ); 272 LocaleDataItem aData = xLocaleData->getLocaleItem(m_aLocale); 273 m_sDecimalSep = aData.decimalSeparator; 274 OSL_ENSURE(m_sDecimalSep.getLength() == 1,"OSingleSelectQueryComposer::OSingleSelectQueryComposer decimal separator is not 1 length"); 275 try 276 { 277 Any aValue; 278 Reference<XInterface> xDs = dbaccess::getDataSource(_xConnection); 279 if ( dbtools::getDataSourceSetting(xDs,static_cast <rtl::OUString> (PROPERTY_BOOLEANCOMPARISONMODE),aValue) ) 280 { 281 OSL_VERIFY( aValue >>= m_nBoolCompareMode ); 282 } 283 Reference< XQueriesSupplier > xQueriesAccess(m_xConnection, UNO_QUERY); 284 if (xQueriesAccess.is()) 285 m_xConnectionQueries = xQueriesAccess->getQueries(); 286 } 287 catch(Exception&) 288 { 289 } 290 } 291 // ------------------------------------------------------------------------- 292 OSingleSelectQueryComposer::~OSingleSelectQueryComposer() 293 { 294 DBG_DTOR(OSingleSelectQueryComposer,NULL); 295 ::std::vector<OPrivateColumns*>::iterator aColIter = m_aColumnsCollection.begin(); 296 ::std::vector<OPrivateColumns*>::iterator aEnd = m_aColumnsCollection.end(); 297 for(;aColIter != aEnd;++aColIter) 298 delete *aColIter; 299 300 ::std::vector<OPrivateTables*>::iterator aTabIter = m_aTablesCollection.begin(); 301 ::std::vector<OPrivateTables*>::iterator aTabEnd = m_aTablesCollection.end(); 302 for(;aTabIter != aTabEnd;++aTabIter) 303 delete *aTabIter; 304 } 305 // ------------------------------------------------------------------------- 306 // OComponentHelper 307 void SAL_CALL OSingleSelectQueryComposer::disposing(void) 308 { 309 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::disposing" ); 310 OSubComponent::disposing(); 311 312 MutexGuard aGuard(m_aMutex); 313 314 resetIterator( m_aSqlIterator, true ); 315 resetIterator( m_aAdditiveIterator, true ); 316 317 m_xConnectionTables = NULL; 318 m_xConnection = NULL; 319 320 clearCurrentCollections(); 321 } 322 IMPLEMENT_FORWARD_XINTERFACE3(OSingleSelectQueryComposer,OSubComponent,OSingleSelectQueryComposer_BASE,OPropertyContainer) 323 IMPLEMENT_SERVICE_INFO1(OSingleSelectQueryComposer,"org.openoffice.comp.dba.OSingleSelectQueryComposer",SERVICE_NAME_SINGLESELECTQUERYCOMPOSER) 324 IMPLEMENT_TYPEPROVIDER3(OSingleSelectQueryComposer,OSubComponent,OSingleSelectQueryComposer_BASE,OPropertyContainer) 325 IMPLEMENT_PROPERTYCONTAINER_DEFAULTS(OSingleSelectQueryComposer) 326 327 // ------------------------------------------------------------------------- 328 // com::sun::star::lang::XUnoTunnel 329 sal_Int64 SAL_CALL OSingleSelectQueryComposer::getSomething( const Sequence< sal_Int8 >& rId ) throw(RuntimeException) 330 { 331 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getSomething" ); 332 if (rId.getLength() == 16 && 0 == rtl_compareMemory(getImplementationId().getConstArray(), rId.getConstArray(), 16 ) ) 333 return reinterpret_cast<sal_Int64>(this); 334 335 return sal_Int64(0); 336 } 337 338 // ------------------------------------------------------------------------- 339 // XSingleSelectQueryAnalyzer 340 ::rtl::OUString SAL_CALL OSingleSelectQueryComposer::getQuery( ) throw(RuntimeException) 341 { 342 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getQuery" ); 343 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 344 ::osl::MutexGuard aGuard( m_aMutex ); 345 346 TGetParseNode F_tmp(&OSQLParseTreeIterator::getParseTree); 347 return getStatementPart(F_tmp,m_aSqlIterator); 348 } 349 350 // ------------------------------------------------------------------------- 351 void SAL_CALL OSingleSelectQueryComposer::setQuery( const ::rtl::OUString& command ) throw(SQLException, RuntimeException) 352 { 353 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "frank.schoenheit@sun.com", "OSingleSelectQueryComposer::setQuery" ); 354 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 355 356 ::osl::MutexGuard aGuard( m_aMutex ); 357 m_nCommandType = CommandType::COMMAND; 358 // first clear the tables and columns 359 clearCurrentCollections(); 360 // now set the new one 361 setQuery_Impl(command); 362 m_sOrignal = command; 363 364 // reset the additive iterator to the same statement 365 parseAndCheck_throwError( m_aSqlParser, m_sOrignal, m_aAdditiveIterator, *this ); 366 367 // we have no "elementary" parts anymore (means filter/groupby/having/order clauses) 368 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 369 m_aElementaryParts[ eLoopParts ] = ::rtl::OUString(); 370 } 371 // ------------------------------------------------------------------------- 372 void SAL_CALL OSingleSelectQueryComposer::setCommand( const ::rtl::OUString& Command,sal_Int32 _nCommandType ) throw(SQLException, RuntimeException) 373 { 374 ::rtl::OUStringBuffer sSQL; 375 switch(_nCommandType) 376 { 377 case CommandType::COMMAND: 378 setElementaryQuery(Command); 379 return; 380 case CommandType::TABLE: 381 if ( m_xConnectionTables->hasByName(Command) ) 382 { 383 sSQL.appendAscii("SELECT * FROM "); 384 Reference< XPropertySet > xTable; 385 try 386 { 387 m_xConnectionTables->getByName( Command ) >>= xTable; 388 } 389 catch(const WrappedTargetException& e) 390 { 391 SQLException e2; 392 if ( e.TargetException >>= e2 ) 393 throw e2; 394 } 395 catch(Exception&) 396 { 397 DBG_UNHANDLED_EXCEPTION(); 398 } 399 400 sSQL.append(dbtools::composeTableNameForSelect(m_xConnection,xTable)); 401 } 402 else 403 { 404 String sMessage( DBACORE_RESSTRING( RID_STR_TABLE_DOES_NOT_EXIST ) ); 405 sMessage.SearchAndReplaceAscii( "$table$", Command ); 406 throwGenericSQLException(sMessage,*this); 407 } 408 break; 409 case CommandType::QUERY: 410 if ( m_xConnectionQueries->hasByName(Command) ) 411 { 412 413 Reference<XPropertySet> xQuery(m_xConnectionQueries->getByName(Command),UNO_QUERY); 414 ::rtl::OUString sCommand; 415 xQuery->getPropertyValue(PROPERTY_COMMAND) >>= sCommand; 416 sSQL.append(sCommand); 417 } 418 else 419 { 420 String sMessage( DBACORE_RESSTRING( RID_STR_QUERY_DOES_NOT_EXIST ) ); 421 sMessage.SearchAndReplaceAscii( "$table$", Command ); 422 throwGenericSQLException(sMessage,*this); 423 } 424 425 break; 426 default: 427 break; 428 } 429 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 430 431 ::osl::MutexGuard aGuard( m_aMutex ); 432 m_nCommandType = _nCommandType; 433 m_sCommand = Command; 434 // first clear the tables and columns 435 clearCurrentCollections(); 436 // now set the new one 437 ::rtl::OUString sCommand = sSQL.makeStringAndClear(); 438 setElementaryQuery(sCommand); 439 m_sOrignal = sCommand; 440 /* 441 // reset the additive iterator to the same statement 442 parseAndCheck_throwError( m_aSqlParser, m_sOrignal, m_aAdditiveIterator, *this ); 443 444 // we have no "elementary" parts anymore (means filter/groupby/having/order clauses) 445 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 446 m_aElementaryParts[ eLoopParts ] = ::rtl::OUString(); 447 */ 448 } 449 // ----------------------------------------------------------------------------- 450 void OSingleSelectQueryComposer::setQuery_Impl( const ::rtl::OUString& command ) 451 { 452 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "frank.schoenheit@sun.com", "OSingleSelectQueryComposer::setQuery_Impl" ); 453 // parse this 454 parseAndCheck_throwError( m_aSqlParser, command, m_aSqlIterator, *this ); 455 456 // strip it from all clauses, to have the pure SELECT statement 457 m_aPureSelectSQL = getPureSelectStatement( m_aSqlIterator.getParseTree(), m_xConnection ); 458 459 // update columns and tables 460 // why? Shouldn't this be done on request only? 461 // otherwise nothing is working anymore :-) 462 // getColumns(); 463 getTables(); 464 } 465 // ----------------------------------------------------------------------------- 466 Sequence< Sequence< PropertyValue > > SAL_CALL OSingleSelectQueryComposer::getStructuredHavingClause( ) throw (RuntimeException) 467 { 468 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getStructuredHavingClause" ); 469 TGetParseNode F_tmp(&OSQLParseTreeIterator::getSimpleHavingTree); 470 return getStructuredCondition(F_tmp); 471 } 472 // ------------------------------------------------------------------------- 473 Sequence< Sequence< PropertyValue > > SAL_CALL OSingleSelectQueryComposer::getStructuredFilter( ) throw(RuntimeException) 474 { 475 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getStructuredFilter" ); 476 TGetParseNode F_tmp(&OSQLParseTreeIterator::getSimpleWhereTree); 477 return getStructuredCondition(F_tmp); 478 } 479 // ----------------------------------------------------------------------------- 480 void SAL_CALL OSingleSelectQueryComposer::appendHavingClauseByColumn( const Reference< XPropertySet >& column, sal_Bool andCriteria,sal_Int32 filterOperator ) throw (SQLException, RuntimeException) 481 { 482 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::appendHavingClauseByColumn" ); 483 ::std::mem_fun1_t<bool,OSingleSelectQueryComposer,::rtl::OUString> F_tmp(&OSingleSelectQueryComposer::implSetHavingClause); 484 setConditionByColumn(column,andCriteria,F_tmp,filterOperator); 485 } 486 // ----------------------------------------------------------------------------- 487 void SAL_CALL OSingleSelectQueryComposer::appendFilterByColumn( const Reference< XPropertySet >& column, sal_Bool andCriteria,sal_Int32 filterOperator ) throw(SQLException, RuntimeException) 488 { 489 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::appendFilterByColumn" ); 490 ::std::mem_fun1_t<bool,OSingleSelectQueryComposer,::rtl::OUString> F_tmp(&OSingleSelectQueryComposer::implSetFilter); 491 setConditionByColumn(column,andCriteria,F_tmp,filterOperator); 492 } 493 // ----------------------------------------------------------------------------- 494 ::rtl::OUString OSingleSelectQueryComposer::impl_getColumnName_throw(const Reference< XPropertySet >& column) 495 { 496 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 497 498 getColumns(); 499 if ( !column.is() 500 || !m_aCurrentColumns[SelectColumns] 501 || !column->getPropertySetInfo()->hasPropertyByName(PROPERTY_NAME) 502 ) 503 { 504 String sError(DBACORE_RESSTRING(RID_STR_COLUMN_UNKNOWN_PROP)); 505 sError.SearchAndReplaceAscii("%value", ::rtl::OUString(PROPERTY_NAME)); 506 SQLException aErr(sError,*this,SQLSTATE_GENERAL,1000,Any() ); 507 throw SQLException(DBACORE_RESSTRING(RID_STR_COLUMN_NOT_VALID),*this,SQLSTATE_GENERAL,1000,makeAny(aErr) ); 508 } 509 510 ::rtl::OUString aName,aNewName; 511 column->getPropertyValue(PROPERTY_NAME) >>= aName; 512 513 if ( !m_xMetaData->supportsOrderByUnrelated() && m_aCurrentColumns[SelectColumns] && !m_aCurrentColumns[SelectColumns]->hasByName(aName)) 514 { 515 String sError(DBACORE_RESSTRING(RID_STR_COLUMN_MUST_VISIBLE)); 516 sError.SearchAndReplaceAscii("%name", aName); 517 throw SQLException(sError,*this,SQLSTATE_GENERAL,1000,Any() ); 518 } 519 520 // filter anhaengen 521 // select ohne where und order by aufbauen 522 ::rtl::OUString aQuote = m_xMetaData->getIdentifierQuoteString(); 523 if ( m_aCurrentColumns[SelectColumns]->hasByName(aName) ) 524 { 525 Reference<XPropertySet> xColumn; 526 m_aCurrentColumns[SelectColumns]->getByName(aName) >>= xColumn; 527 OSL_ENSURE(xColumn->getPropertySetInfo()->hasPropertyByName(PROPERTY_REALNAME),"Property REALNAME not available!"); 528 OSL_ENSURE(xColumn->getPropertySetInfo()->hasPropertyByName(PROPERTY_TABLENAME),"Property TABLENAME not available!"); 529 OSL_ENSURE(xColumn->getPropertySetInfo()->hasPropertyByName(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("Function"))),"Property FUNCTION not available!"); 530 531 ::rtl::OUString sRealName,sTableName; 532 xColumn->getPropertyValue(PROPERTY_REALNAME) >>= sRealName; 533 xColumn->getPropertyValue(PROPERTY_TABLENAME) >>= sTableName; 534 sal_Bool bFunction = sal_False; 535 xColumn->getPropertyValue(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("Function"))) >>= bFunction; 536 if ( sRealName == aName ) 537 { 538 if ( bFunction ) 539 aNewName = aName; 540 else 541 { 542 if(sTableName.indexOf('.',0) != -1) 543 { 544 ::rtl::OUString aCatlog,aSchema,aTable; 545 ::dbtools::qualifiedNameComponents(m_xMetaData,sTableName,aCatlog,aSchema,aTable,::dbtools::eInDataManipulation); 546 sTableName = ::dbtools::composeTableName( m_xMetaData, aCatlog, aSchema, aTable, sal_True, ::dbtools::eInDataManipulation ); 547 } 548 else 549 sTableName = ::dbtools::quoteName(aQuote,sTableName); 550 551 aNewName = sTableName; 552 aNewName += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(".")); 553 aNewName += ::dbtools::quoteName(aQuote,sRealName); 554 } 555 } 556 else 557 aNewName = ::dbtools::quoteName(aQuote,aName); 558 } 559 else 560 aNewName = getTableAlias(column) + ::dbtools::quoteName(aQuote,aName); 561 return aNewName; 562 } 563 // ------------------------------------------------------------------------- 564 void SAL_CALL OSingleSelectQueryComposer::appendOrderByColumn( const Reference< XPropertySet >& column, sal_Bool ascending ) throw(SQLException, RuntimeException) 565 { 566 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::appendOrderByColumn" ); 567 ::osl::MutexGuard aGuard( m_aMutex ); 568 ::rtl::OUString sColumnName( impl_getColumnName_throw(column) ); 569 ::rtl::OUString sOrder = getOrder(); 570 if ( (sOrder.getLength() != 0) && sColumnName.getLength() ) 571 sOrder += COMMA; 572 sOrder += sColumnName; 573 if ( !ascending && sColumnName.getLength() ) 574 sOrder += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" DESC ")); 575 576 setOrder(sOrder); 577 } 578 579 // ------------------------------------------------------------------------- 580 void SAL_CALL OSingleSelectQueryComposer::appendGroupByColumn( const Reference< XPropertySet >& column) throw(SQLException, RuntimeException) 581 { 582 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::appendGroupByColumn" ); 583 ::osl::MutexGuard aGuard( m_aMutex ); 584 ::rtl::OUString sColumnName( impl_getColumnName_throw(column) ); 585 OrderCreator aComposer; 586 aComposer.append( getGroup() ); 587 aComposer.append( sColumnName ); 588 setGroup( aComposer.getComposedAndClear() ); 589 } 590 // ------------------------------------------------------------------------- 591 ::rtl::OUString OSingleSelectQueryComposer::composeStatementFromParts( const ::std::vector< ::rtl::OUString >& _rParts ) 592 { 593 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::composeStatementFromParts" ); 594 DBG_ASSERT( _rParts.size() == (size_t)SQLPartCount, "OSingleSelectQueryComposer::composeStatementFromParts: invalid parts array!" ); 595 596 ::rtl::OUStringBuffer aSql( m_aPureSelectSQL ); 597 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 598 if ( _rParts[ eLoopParts ].getLength() ) 599 { 600 aSql.append( getKeyword( eLoopParts ) ); 601 aSql.append( _rParts[ eLoopParts ] ); 602 } 603 604 return aSql.makeStringAndClear(); 605 } 606 607 // ------------------------------------------------------------------------- 608 ::rtl::OUString SAL_CALL OSingleSelectQueryComposer::getElementaryQuery() throw (::com::sun::star::uno::RuntimeException) 609 { 610 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getElementaryQuery" ); 611 return composeStatementFromParts( m_aElementaryParts ); 612 } 613 614 // ------------------------------------------------------------------------- 615 void SAL_CALL OSingleSelectQueryComposer::setElementaryQuery( const ::rtl::OUString& _rElementary ) throw (::com::sun::star::sdbc::SQLException, ::com::sun::star::uno::RuntimeException) 616 { 617 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "frank.schoenheit@sun.com", "OSingleSelectQueryComposer::setElementaryQuery" ); 618 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 619 ::osl::MutexGuard aGuard( m_aMutex ); 620 621 // remember the 4 current "additive" clauses 622 ::std::vector< ::rtl::OUString > aAdditiveClauses( SQLPartCount ); 623 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 624 aAdditiveClauses[ eLoopParts ] = getSQLPart( eLoopParts, m_aAdditiveIterator, sal_False ); 625 626 // clear the tables and columns 627 clearCurrentCollections(); 628 // set and parse the new query 629 setQuery_Impl( _rElementary ); 630 631 // get the 4 elementary parts of the statement 632 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 633 m_aElementaryParts[ eLoopParts ] = getSQLPart( eLoopParts, m_aSqlIterator, sal_False ); 634 635 // reset the the AdditiveIterator: m_aPureSelectSQL may have changed 636 try 637 { 638 parseAndCheck_throwError( m_aSqlParser, composeStatementFromParts( aAdditiveClauses ), m_aAdditiveIterator, *this ); 639 } 640 catch( const Exception& e ) 641 { 642 (void)e; 643 DBG_ERROR( "OSingleSelectQueryComposer::setElementaryQuery: there should be no error anymore for the additive statement!" ); 644 // every part of the additive statement should have passed other tests already, and should not 645 // be able to cause any errors ... me thinks 646 } 647 } 648 649 // ------------------------------------------------------------------------- 650 namespace 651 { 652 ::rtl::OUString getComposedClause( const ::rtl::OUString _rElementaryClause, const ::rtl::OUString _rAdditionalClause, 653 TokenComposer& _rComposer, const ::rtl::OUString _rKeyword ) 654 { 655 _rComposer.clear(); 656 _rComposer.append( _rElementaryClause ); 657 _rComposer.append( _rAdditionalClause ); 658 ::rtl::OUString sComposed = _rComposer.getComposedAndClear(); 659 if ( sComposed.getLength() ) 660 sComposed = _rKeyword + sComposed; 661 return sComposed; 662 } 663 } 664 665 // ------------------------------------------------------------------------- 666 void OSingleSelectQueryComposer::setSingleAdditiveClause( SQLPart _ePart, const ::rtl::OUString& _rClause ) 667 { 668 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setSingleAdditiveClause" ); 669 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 670 ::osl::MutexGuard aGuard( m_aMutex ); 671 672 // if nothing is changed, do nothing 673 if ( getSQLPart( _ePart, m_aAdditiveIterator, sal_False ) == _rClause ) 674 return; 675 676 // collect the 4 single parts as they're currently set 677 ::std::vector< ::rtl::OUString > aClauses; 678 aClauses.reserve( (size_t)SQLPartCount ); 679 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 680 aClauses.push_back( getSQLPart( eLoopParts, m_aSqlIterator, sal_True ) ); 681 682 // overwrite the one part in question here 683 ::std::auto_ptr< TokenComposer > pComposer; 684 if ( ( _ePart == Where ) || ( _ePart == Having ) ) 685 pComposer.reset( new FilterCreator ); 686 else 687 pComposer.reset( new OrderCreator ); 688 aClauses[ _ePart ] = getComposedClause( m_aElementaryParts[ _ePart ], _rClause, 689 *pComposer, getKeyword( _ePart ) ); 690 691 // construct the complete statement 692 ::rtl::OUStringBuffer aSql(m_aPureSelectSQL); 693 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 694 aSql.append(aClauses[ eLoopParts ]); 695 696 // set the query 697 setQuery_Impl(aSql.makeStringAndClear()); 698 699 // clear column collections which (might) have changed 700 clearColumns( ParameterColumns ); 701 if ( _ePart == Order ) 702 clearColumns( OrderColumns ); 703 if ( _ePart == Group ) 704 clearColumns( GroupByColumns ); 705 706 // also, since the "additive filter" change, we need to rebuild our "additive" statement 707 aSql = m_aPureSelectSQL; 708 // again, first get all the old additive parts 709 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 710 aClauses[ eLoopParts ] = getSQLPart( eLoopParts, m_aAdditiveIterator, sal_True ); 711 // then overwrite the one in question 712 aClauses[ _ePart ] = getComposedClause( ::rtl::OUString(), _rClause, *pComposer, getKeyword( _ePart ) ); 713 // and parse it, so that m_aAdditiveIterator is up to date 714 for ( SQLPart eLoopParts = Where; eLoopParts != SQLPartCount; incSQLPart( eLoopParts ) ) 715 aSql.append(aClauses[ eLoopParts ]); 716 try 717 { 718 parseAndCheck_throwError( m_aSqlParser, aSql.makeStringAndClear(), m_aAdditiveIterator, *this ); 719 } 720 catch( const Exception& e ) 721 { 722 (void)e; 723 DBG_ERROR( "OSingleSelectQueryComposer::setSingleAdditiveClause: there should be no error anymore for the additive statement!" ); 724 // every part of the additive statement should have passed other tests already, and should not 725 // be able to cause any errors ... me thinks 726 } 727 } 728 729 // ------------------------------------------------------------------------- 730 void SAL_CALL OSingleSelectQueryComposer::setFilter( const ::rtl::OUString& filter ) throw(SQLException, RuntimeException) 731 { 732 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setFilter" ); 733 setSingleAdditiveClause( Where, filter ); 734 } 735 736 // ------------------------------------------------------------------------- 737 void SAL_CALL OSingleSelectQueryComposer::setOrder( const ::rtl::OUString& order ) throw(SQLException, RuntimeException) 738 { 739 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setOrder" ); 740 setSingleAdditiveClause( Order, order ); 741 } 742 // ----------------------------------------------------------------------------- 743 void SAL_CALL OSingleSelectQueryComposer::setGroup( const ::rtl::OUString& group ) throw (SQLException, RuntimeException) 744 { 745 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setGroup" ); 746 setSingleAdditiveClause( Group, group ); 747 } 748 // ------------------------------------------------------------------------- 749 void SAL_CALL OSingleSelectQueryComposer::setHavingClause( const ::rtl::OUString& filter ) throw(SQLException, RuntimeException) 750 { 751 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setHavingClause" ); 752 setSingleAdditiveClause( Having, filter ); 753 } 754 755 // ------------------------------------------------------------------------- 756 // XTablesSupplier 757 Reference< XNameAccess > SAL_CALL OSingleSelectQueryComposer::getTables( ) throw(RuntimeException) 758 { 759 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getTables" ); 760 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 761 762 ::osl::MutexGuard aGuard( m_aMutex ); 763 if ( !m_pTables ) 764 { 765 const OSQLTables& aTables = m_aSqlIterator.getTables(); 766 ::std::vector< ::rtl::OUString> aNames; 767 OSQLTables::const_iterator aEnd = aTables.end(); 768 for(OSQLTables::const_iterator aIter = aTables.begin(); aIter != aEnd;++aIter) 769 aNames.push_back(aIter->first); 770 771 m_pTables = new OPrivateTables(aTables,m_xMetaData->supportsMixedCaseQuotedIdentifiers(),*this,m_aMutex,aNames); 772 } 773 774 return m_pTables; 775 } 776 // ------------------------------------------------------------------------- 777 // XColumnsSupplier 778 Reference< XNameAccess > SAL_CALL OSingleSelectQueryComposer::getColumns( ) throw(RuntimeException) 779 { 780 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getColumns" ); 781 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 782 ::osl::MutexGuard aGuard( m_aMutex ); 783 if ( !!m_aCurrentColumns[SelectColumns] ) 784 return m_aCurrentColumns[SelectColumns]; 785 786 ::std::vector< ::rtl::OUString> aNames; 787 ::vos::ORef< OSQLColumns> aSelectColumns; 788 sal_Bool bCase = sal_True; 789 Reference< XNameAccess> xQueryColumns; 790 if ( m_nCommandType == CommandType::QUERY ) 791 { 792 Reference<XColumnsSupplier> xSup(m_xConnectionQueries->getByName(m_sCommand),UNO_QUERY); 793 if(xSup.is()) 794 xQueryColumns = xSup->getColumns(); 795 } 796 797 do { 798 799 try 800 { 801 SharedUNOComponent< XStatement, DisposableComponent > xStatement; 802 SharedUNOComponent< XPreparedStatement, DisposableComponent > xPreparedStatement; 803 804 bCase = m_xMetaData->supportsMixedCaseQuotedIdentifiers(); 805 aSelectColumns = m_aSqlIterator.getSelectColumns(); 806 807 ::rtl::OUStringBuffer aSQL; 808 aSQL.append( m_aPureSelectSQL ); 809 aSQL.append( STR_WHERE ); 810 811 // preserve the original WHERE clause 812 // #i102234# / 2009-06-02 / frank.schoenheit@sun.com 813 ::rtl::OUString sOriginalWhereClause = getSQLPart( Where, m_aSqlIterator, sal_False ); 814 if ( sOriginalWhereClause.getLength() ) 815 { 816 aSQL.appendAscii( " ( 0 = 1 ) AND ( " ); 817 aSQL.append( sOriginalWhereClause ); 818 aSQL.appendAscii( " ) " ); 819 } 820 else 821 { 822 aSQL.appendAscii( " ( 0 = 1 ) " ); 823 } 824 825 ::rtl::OUString sGroupBy = getSQLPart( Group, m_aSqlIterator, sal_True ); 826 if ( sGroupBy.getLength() ) 827 aSQL.append( sGroupBy ); 828 829 ::rtl::OUString sSQL( aSQL.makeStringAndClear() ); 830 // normalize the statement so that it doesn't contain any application-level features anymore 831 ::rtl::OUString sError; 832 const ::std::auto_ptr< OSQLParseNode > pStatementTree( m_aSqlParser.parseTree( sError, sSQL, false ) ); 833 DBG_ASSERT( pStatementTree.get(), "OSingleSelectQueryComposer::getColumns: could not parse the column retrieval statement!" ); 834 if ( pStatementTree.get() ) 835 if ( !pStatementTree->parseNodeToExecutableStatement( sSQL, m_xConnection, m_aSqlParser, NULL ) ) 836 break; 837 838 Reference< XResultSetMetaData > xResultSetMeta; 839 Reference< XResultSetMetaDataSupplier > xResMetaDataSup; 840 try 841 { 842 xPreparedStatement.set( m_xConnection->prepareStatement( sSQL ), UNO_QUERY_THROW ); 843 xResMetaDataSup.set( xPreparedStatement, UNO_QUERY_THROW ); 844 xResultSetMeta.set( xResMetaDataSup->getMetaData(), UNO_QUERY_THROW ); 845 } 846 catch( const Exception& ) { } 847 848 try 849 { 850 if ( !xResultSetMeta.is() ) 851 { 852 xStatement.reset( Reference< XStatement >( m_xConnection->createStatement(), UNO_QUERY_THROW ) ); 853 Reference< XPropertySet > xStatementProps( xStatement, UNO_QUERY_THROW ); 854 try { xStatementProps->setPropertyValue( PROPERTY_ESCAPE_PROCESSING, makeAny( sal_False ) ); } 855 catch ( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } 856 xResMetaDataSup.set( xStatement->executeQuery( sSQL ), UNO_QUERY_THROW ); 857 xResultSetMeta.set( xResMetaDataSup->getMetaData(), UNO_QUERY_THROW ); 858 } 859 } 860 catch( const Exception& ) 861 { 862 //@see issue http://qa.openoffice.org/issues/show_bug.cgi?id=110111 863 // access returns a different order of column names when executing select * from 864 // and asking the columns from the metadata. 865 Reference< XParameters > xParameters( xPreparedStatement, UNO_QUERY_THROW ); 866 Reference< XIndexAccess > xPara = getParameters(); 867 for(sal_Int32 i = 1;i <= xPara->getCount();++i) 868 xParameters->setNull(i,DataType::VARCHAR); 869 xResMetaDataSup.set(xPreparedStatement->executeQuery(), UNO_QUERY_THROW ); 870 xResultSetMeta.set( xResMetaDataSup->getMetaData(), UNO_QUERY_THROW ); 871 } 872 873 if ( aSelectColumns->get().empty() ) 874 { 875 // This is a valid case. If we can syntactically parse the query, but not semantically 876 // (e.g. because it is based on a table we do not know), then there will be no SelectColumns 877 aSelectColumns = ::connectivity::parse::OParseColumn::createColumnsForResultSet( xResultSetMeta, m_xMetaData ,xQueryColumns); 878 break; 879 } 880 881 const ::comphelper::UStringMixEqual aCaseCompare( bCase ); 882 const ::comphelper::TStringMixEqualFunctor aCaseCompareFunctor( bCase ); 883 typedef ::std::set< size_t > SizeTSet; 884 SizeTSet aUsedSelectColumns; 885 ::connectivity::parse::OParseColumn::StringMap aColumnNames; 886 887 sal_Int32 nCount = xResultSetMeta->getColumnCount(); 888 OSL_ENSURE( (size_t) nCount == aSelectColumns->get().size(), "OSingleSelectQueryComposer::getColumns: inconsistent column counts, this might result in wrong columns!" ); 889 for(sal_Int32 i=1;i<=nCount;++i) 890 { 891 ::rtl::OUString sColumnName = xResultSetMeta->getColumnName(i); 892 ::rtl::OUString sColumnLabel; 893 if ( xQueryColumns.is() && xQueryColumns->hasByName(sColumnName) ) 894 { 895 Reference<XPropertySet> xQueryColumn(xQueryColumns->getByName(sColumnName),UNO_QUERY_THROW); 896 xQueryColumn->getPropertyValue(PROPERTY_LABEL) >>= sColumnLabel; 897 } 898 else 899 sColumnLabel = xResultSetMeta->getColumnLabel(i); 900 sal_Bool bFound = sal_False; 901 OSQLColumns::Vector::const_iterator aFind = ::connectivity::find(aSelectColumns->get().begin(),aSelectColumns->get().end(),sColumnLabel,aCaseCompare); 902 size_t nFoundSelectColumnPos = aFind - aSelectColumns->get().begin(); 903 if ( aFind != aSelectColumns->get().end() ) 904 { 905 if ( aUsedSelectColumns.find( nFoundSelectColumnPos ) != aUsedSelectColumns.end() ) 906 { // we found a column name which exists twice 907 // so we start after the first found 908 do 909 { 910 aFind = ::connectivity::findRealName(++aFind,aSelectColumns->get().end(),sColumnName,aCaseCompare); 911 nFoundSelectColumnPos = aFind - aSelectColumns->get().begin(); 912 } 913 while ( ( aUsedSelectColumns.find( nFoundSelectColumnPos ) != aUsedSelectColumns.end() ) 914 && ( aFind != aSelectColumns->get().end() ) 915 ); 916 } 917 if ( aFind != aSelectColumns->get().end() ) 918 { 919 (*aFind)->getPropertyValue(PROPERTY_NAME) >>= sColumnName; 920 aUsedSelectColumns.insert( nFoundSelectColumnPos ); 921 aNames.push_back(sColumnName); 922 bFound = sal_True; 923 } 924 } 925 926 if ( bFound ) 927 continue; 928 929 OSQLColumns::Vector::const_iterator aRealFind = ::connectivity::findRealName( 930 aSelectColumns->get().begin(), aSelectColumns->get().end(), sColumnName, aCaseCompare ); 931 932 if ( i > static_cast< sal_Int32>( aSelectColumns->get().size() ) ) 933 { 934 aSelectColumns->get().push_back( 935 ::connectivity::parse::OParseColumn::createColumnForResultSet( xResultSetMeta, m_xMetaData, i ,aColumnNames) 936 ); 937 OSL_ENSURE( aSelectColumns->get().size() == (size_t)i, "OSingleSelectQueryComposer::getColumns: inconsistency!" ); 938 } 939 else if ( aRealFind == aSelectColumns->get().end() ) 940 { 941 // we can now only look if we found it under the realname propertery 942 // here we have to make the assumption that the position is correct 943 OSQLColumns::Vector::iterator aFind2 = aSelectColumns->get().begin() + i-1; 944 Reference<XPropertySet> xProp(*aFind2,UNO_QUERY); 945 if ( !xProp.is() || !xProp->getPropertySetInfo()->hasPropertyByName( PROPERTY_REALNAME ) ) 946 continue; 947 948 ::connectivity::parse::OParseColumn* pColumn = new ::connectivity::parse::OParseColumn(xProp,bCase); 949 pColumn->setFunction(::comphelper::getBOOL(xProp->getPropertyValue(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("Function"))))); 950 pColumn->setAggregateFunction(::comphelper::getBOOL(xProp->getPropertyValue(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("AggregateFunction"))))); 951 952 ::rtl::OUString sRealName; 953 xProp->getPropertyValue(PROPERTY_REALNAME) >>= sRealName; 954 ::std::vector< ::rtl::OUString>::iterator aFindName; 955 if ( !sColumnName.getLength() ) 956 xProp->getPropertyValue(PROPERTY_NAME) >>= sColumnName; 957 958 959 aFindName = ::std::find_if(aNames.begin(),aNames.end(),::std::bind2nd(aCaseCompareFunctor,sColumnName)); 960 sal_Int32 j = 0; 961 while ( aFindName != aNames.end() ) 962 { 963 sColumnName += ::rtl::OUString::valueOf(++j); 964 aFindName = ::std::find_if(aNames.begin(),aNames.end(),::std::bind2nd(aCaseCompareFunctor,sColumnName)); 965 } 966 967 pColumn->setName(sColumnName); 968 pColumn->setRealName(sRealName); 969 pColumn->setTableName(::comphelper::getString(xProp->getPropertyValue(PROPERTY_TABLENAME))); 970 971 (aSelectColumns->get())[i-1] = pColumn; 972 } 973 else 974 continue; 975 976 aUsedSelectColumns.insert( (size_t)(i - 1) ); 977 aNames.push_back( sColumnName ); 978 } 979 } 980 catch(const Exception&) 981 { 982 } 983 984 } while ( false ); 985 986 if ( aNames.empty() ) 987 m_aCurrentColumns[ SelectColumns ] = OPrivateColumns::createWithIntrinsicNames( aSelectColumns, bCase, *this, m_aMutex ); 988 else 989 m_aCurrentColumns[ SelectColumns ] = new OPrivateColumns( aSelectColumns, bCase, *this, m_aMutex, aNames ); 990 991 return m_aCurrentColumns[SelectColumns]; 992 } 993 // ------------------------------------------------------------------------- 994 sal_Bool OSingleSelectQueryComposer::setORCriteria(OSQLParseNode* pCondition, OSQLParseTreeIterator& _rIterator, 995 ::std::vector< ::std::vector < PropertyValue > >& rFilters, const Reference< ::com::sun::star::util::XNumberFormatter > & xFormatter) const 996 { 997 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setORCriteria" ); 998 // Runde Klammern um den Ausdruck 999 if (pCondition->count() == 3 && 1000 SQL_ISPUNCTUATION(pCondition->getChild(0),"(") && 1001 SQL_ISPUNCTUATION(pCondition->getChild(2),")")) 1002 { 1003 return setORCriteria(pCondition->getChild(1), _rIterator, rFilters, xFormatter); 1004 } 1005 // oder Verknuepfung 1006 // a searchcondition can only look like this: search_condition SQL_TOKEN_OR boolean_term 1007 else if (SQL_ISRULE(pCondition,search_condition)) 1008 { 1009 sal_Bool bResult = sal_True; 1010 for (int i = 0; bResult && i < 3; i+=2) 1011 { 1012 // Ist das erste Element wieder eine OR-Verknuepfung? 1013 // Dann rekursiv absteigen ... 1014 //if (!i && SQL_ISRULE(pCondition->getChild(i),search_condition)) 1015 if (SQL_ISRULE(pCondition->getChild(i),search_condition)) 1016 bResult = setORCriteria(pCondition->getChild(i), _rIterator, rFilters, xFormatter); 1017 else 1018 { 1019 rFilters.push_back( ::std::vector < PropertyValue >()); 1020 bResult = setANDCriteria(pCondition->getChild(i), _rIterator, rFilters[rFilters.size() - 1], xFormatter); 1021 } 1022 } 1023 return bResult; 1024 } 1025 else 1026 { 1027 rFilters.push_back(::std::vector < PropertyValue >()); 1028 return setANDCriteria(pCondition, _rIterator, rFilters[rFilters.size() - 1], xFormatter); 1029 } 1030 } 1031 1032 //-------------------------------------------------------------------------------------------------- 1033 sal_Bool OSingleSelectQueryComposer::setANDCriteria( OSQLParseNode * pCondition, 1034 OSQLParseTreeIterator& _rIterator, ::std::vector < PropertyValue >& rFilter, const Reference< XNumberFormatter > & xFormatter) const 1035 { 1036 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setANDCriteria" ); 1037 // Runde Klammern 1038 if (SQL_ISRULE(pCondition,boolean_primary)) 1039 { 1040 // this should not occur 1041 DBG_ERROR("boolean_primary in And-Criteria"); 1042 return sal_False; 1043 } 1044 // Das erste Element ist (wieder) eine AND-Verknuepfung 1045 else if ( SQL_ISRULE(pCondition,boolean_term) && pCondition->count() == 3 ) 1046 { 1047 return setANDCriteria(pCondition->getChild(0), _rIterator, rFilter, xFormatter) && 1048 setANDCriteria(pCondition->getChild(2), _rIterator, rFilter, xFormatter); 1049 } 1050 else if (SQL_ISRULE(pCondition, comparison_predicate)) 1051 { 1052 return setComparsionPredicate(pCondition,_rIterator,rFilter,xFormatter); 1053 } 1054 else if (SQL_ISRULE(pCondition,like_predicate) || 1055 SQL_ISRULE(pCondition,test_for_null) || 1056 SQL_ISRULE(pCondition,in_predicate) || 1057 SQL_ISRULE(pCondition,all_or_any_predicate) || 1058 SQL_ISRULE(pCondition,between_predicate)) 1059 { 1060 if (SQL_ISRULE(pCondition->getChild(0), column_ref)) 1061 { 1062 PropertyValue aItem; 1063 ::rtl::OUString aValue; 1064 ::rtl::OUString aColumnName; 1065 1066 1067 // pCondition->parseNodeToStr(aValue,m_xMetaData, xFormatter, m_aLocale,static_cast<sal_Char>(m_sDecimalSep.toChar())); 1068 pCondition->parseNodeToStr( aValue, m_xConnection, NULL ); 1069 // pCondition->getChild(0)->parseNodeToStr(aColumnName,m_xMetaData, xFormatter, m_aLocale,static_cast<sal_Char>(m_sDecimalSep.toChar())); 1070 pCondition->getChild(0)->parseNodeToStr( aColumnName, m_xConnection, NULL ); 1071 1072 // don't display the column name 1073 aValue = aValue.copy(aColumnName.getLength()); 1074 aValue = aValue.trim(); 1075 1076 aItem.Name = getColumnName(pCondition->getChild(0),_rIterator); 1077 aItem.Value <<= aValue; 1078 aItem.Handle = 0; // just to know that this is not one the known ones 1079 if ( SQL_ISRULE(pCondition,like_predicate) ) 1080 { 1081 if ( SQL_ISTOKEN(pCondition->getChild(1)->getChild(0),NOT) ) 1082 aItem.Handle = SQLFilterOperator::NOT_LIKE; 1083 else 1084 aItem.Handle = SQLFilterOperator::LIKE; 1085 } 1086 else if (SQL_ISRULE(pCondition,test_for_null)) 1087 { 1088 if (SQL_ISTOKEN(pCondition->getChild(1)->getChild(1),NOT) ) 1089 aItem.Handle = SQLFilterOperator::NOT_SQLNULL; 1090 else 1091 aItem.Handle = SQLFilterOperator::SQLNULL; 1092 } 1093 else if (SQL_ISRULE(pCondition,in_predicate)) 1094 { 1095 OSL_ENSURE( false, "OSingleSelectQueryComposer::setANDCriteria: in_predicate not implemented!" ); 1096 } 1097 else if (SQL_ISRULE(pCondition,all_or_any_predicate)) 1098 { 1099 OSL_ENSURE( false, "OSingleSelectQueryComposer::setANDCriteria: all_or_any_predicate not implemented!" ); 1100 } 1101 else if (SQL_ISRULE(pCondition,between_predicate)) 1102 { 1103 OSL_ENSURE( false, "OSingleSelectQueryComposer::setANDCriteria: between_predicate not implemented!" ); 1104 } 1105 1106 rFilter.push_back(aItem); 1107 } 1108 else 1109 return sal_False; 1110 } 1111 else if (SQL_ISRULE(pCondition,existence_test) || 1112 SQL_ISRULE(pCondition,unique_test)) 1113 { 1114 // this couldn't be handled here, too complex 1115 // as we need a field name 1116 return sal_False; 1117 } 1118 else 1119 return sal_False; 1120 1121 return sal_True; 1122 } 1123 // ----------------------------------------------------------------------------- 1124 sal_Int32 OSingleSelectQueryComposer::getPredicateType(OSQLParseNode * _pPredicate) const 1125 { 1126 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getPredicateType" ); 1127 sal_Int32 nPredicate = SQLFilterOperator::EQUAL; 1128 switch (_pPredicate->getNodeType()) 1129 { 1130 case SQL_NODE_EQUAL: 1131 nPredicate = SQLFilterOperator::EQUAL; 1132 break; 1133 case SQL_NODE_NOTEQUAL: 1134 nPredicate = SQLFilterOperator::NOT_EQUAL; 1135 break; 1136 case SQL_NODE_LESS: 1137 nPredicate = SQLFilterOperator::LESS; 1138 break; 1139 case SQL_NODE_LESSEQ: 1140 nPredicate = SQLFilterOperator::LESS_EQUAL; 1141 break; 1142 case SQL_NODE_GREAT: 1143 nPredicate = SQLFilterOperator::GREATER; 1144 break; 1145 case SQL_NODE_GREATEQ: 1146 nPredicate = SQLFilterOperator::GREATER_EQUAL; 1147 break; 1148 default: 1149 OSL_ENSURE(0,"Wrong NodeType!"); 1150 } 1151 return nPredicate; 1152 } 1153 //------------------------------------------------------------------------------ 1154 sal_Bool OSingleSelectQueryComposer::setComparsionPredicate(OSQLParseNode * pCondition, OSQLParseTreeIterator& _rIterator, 1155 ::std::vector < PropertyValue >& rFilter, const Reference< ::com::sun::star::util::XNumberFormatter > & xFormatter) const 1156 { 1157 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setComparsionPredicate" ); 1158 DBG_ASSERT(SQL_ISRULE(pCondition, comparison_predicate),"setComparsionPredicate: pCondition ist kein ComparsionPredicate"); 1159 if (SQL_ISRULE(pCondition->getChild(0), column_ref) || 1160 SQL_ISRULE(pCondition->getChild(pCondition->count()-1), column_ref)) 1161 { 1162 PropertyValue aItem; 1163 ::rtl::OUString aValue; 1164 sal_uInt32 nPos; 1165 if (SQL_ISRULE(pCondition->getChild(0), column_ref)) 1166 { 1167 nPos = 0; 1168 sal_uInt32 i=1; 1169 1170 aItem.Handle = getPredicateType(pCondition->getChild(i)); 1171 // don't display the equal 1172 if (pCondition->getChild(i)->getNodeType() == SQL_NODE_EQUAL) 1173 i++; 1174 1175 // go forward 1176 for (;i < pCondition->count();i++) 1177 pCondition->getChild(i)->parseNodeToPredicateStr( 1178 aValue, m_xConnection, xFormatter, m_aLocale, static_cast<sal_Char>(m_sDecimalSep.toChar() ) ); 1179 } 1180 else if (SQL_ISRULE(pCondition->getChild(pCondition->count()-1), column_ref)) 1181 { 1182 nPos = pCondition->count()-1; 1183 1184 sal_Int32 i = pCondition->count() - 2; 1185 switch (pCondition->getChild(i)->getNodeType()) 1186 { 1187 case SQL_NODE_EQUAL: 1188 // don't display the equal 1189 i--; 1190 aItem.Handle = SQLFilterOperator::EQUAL; 1191 break; 1192 case SQL_NODE_NOTEQUAL: 1193 i--; 1194 aItem.Handle = SQLFilterOperator::NOT_EQUAL; 1195 break; 1196 case SQL_NODE_LESS: 1197 // take the opposite as we change the order 1198 i--; 1199 aValue = ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(">=")); 1200 aItem.Handle = SQLFilterOperator::GREATER_EQUAL; 1201 break; 1202 case SQL_NODE_LESSEQ: 1203 // take the opposite as we change the order 1204 i--; 1205 aValue = ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(">")); 1206 aItem.Handle = SQLFilterOperator::GREATER; 1207 break; 1208 case SQL_NODE_GREAT: 1209 // take the opposite as we change the order 1210 i--; 1211 aValue = ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("<=")); 1212 aItem.Handle = SQLFilterOperator::LESS_EQUAL; 1213 break; 1214 case SQL_NODE_GREATEQ: 1215 // take the opposite as we change the order 1216 i--; 1217 aValue = ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("<")); 1218 aItem.Handle = SQLFilterOperator::LESS; 1219 break; 1220 default: 1221 break; 1222 } 1223 1224 // go backward 1225 for (; i >= 0; i--) 1226 pCondition->getChild(i)->parseNodeToPredicateStr( 1227 aValue, m_xConnection, xFormatter, m_aLocale, static_cast<sal_Char>( m_sDecimalSep.toChar() ) ); 1228 } 1229 else 1230 return sal_False; 1231 1232 aItem.Name = getColumnName(pCondition->getChild(nPos),_rIterator); 1233 aItem.Value <<= aValue; 1234 rFilter.push_back(aItem); 1235 } 1236 else if (SQL_ISRULE(pCondition->getChild(0), set_fct_spec ) || 1237 SQL_ISRULE(pCondition->getChild(0), general_set_fct)) 1238 { 1239 PropertyValue aItem; 1240 ::rtl::OUString aValue; 1241 ::rtl::OUString aColumnName; 1242 1243 pCondition->getChild(2)->parseNodeToPredicateStr(aValue, m_xConnection, xFormatter, m_aLocale, static_cast<sal_Char>( m_sDecimalSep.toChar() ) ); 1244 pCondition->getChild(0)->parseNodeToPredicateStr( aColumnName, m_xConnection, xFormatter, m_aLocale, static_cast<sal_Char>( m_sDecimalSep .toChar() ) ); 1245 1246 aItem.Name = getColumnName(pCondition->getChild(0),_rIterator); 1247 aItem.Value <<= aValue; 1248 aItem.Handle = getPredicateType(pCondition->getChild(1)); 1249 rFilter.push_back(aItem); 1250 } 1251 else // kann sich nur um einen Expr. Ausdruck handeln 1252 { 1253 PropertyValue aItem; 1254 ::rtl::OUString aName, aValue; 1255 1256 OSQLParseNode *pLhs = pCondition->getChild(0); 1257 OSQLParseNode *pRhs = pCondition->getChild(2); 1258 1259 // Feldnamen 1260 sal_uInt16 i; 1261 for (i=0;i< pLhs->count();i++) 1262 pLhs->getChild(i)->parseNodeToPredicateStr( aName, m_xConnection, xFormatter, m_aLocale, static_cast<sal_Char>( m_sDecimalSep.toChar() ) ); 1263 1264 // Kriterium 1265 aItem.Handle = getPredicateType(pCondition->getChild(1)); 1266 aValue = pCondition->getChild(1)->getTokenValue(); 1267 for(i=0;i< pRhs->count();i++) 1268 pRhs->getChild(i)->parseNodeToPredicateStr(aValue, m_xConnection, xFormatter, m_aLocale, static_cast<sal_Char>( m_sDecimalSep.toChar() ) ); 1269 1270 aItem.Name = aName; 1271 aItem.Value <<= aValue; 1272 rFilter.push_back(aItem); 1273 } 1274 return sal_True; 1275 } 1276 // functions for analysing SQL 1277 //-------------------------------------------------------------------------------------------------- 1278 ::rtl::OUString OSingleSelectQueryComposer::getColumnName( ::connectivity::OSQLParseNode* pColumnRef, OSQLParseTreeIterator& _rIterator ) const 1279 { 1280 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getColumnName" ); 1281 ::rtl::OUString aTableRange, aColumnName; 1282 _rIterator.getColumnRange(pColumnRef,aColumnName,aTableRange); 1283 return aColumnName; 1284 } 1285 //------------------------------------------------------------------------------ 1286 ::rtl::OUString SAL_CALL OSingleSelectQueryComposer::getFilter( ) throw(RuntimeException) 1287 { 1288 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getFilter" ); 1289 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1290 ::osl::MutexGuard aGuard( m_aMutex ); 1291 return getSQLPart(Where,m_aAdditiveIterator,sal_False); 1292 } 1293 // ------------------------------------------------------------------------- 1294 ::rtl::OUString SAL_CALL OSingleSelectQueryComposer::getOrder( ) throw(RuntimeException) 1295 { 1296 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getOrder" ); 1297 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1298 ::osl::MutexGuard aGuard( m_aMutex ); 1299 return getSQLPart(Order,m_aAdditiveIterator,sal_False); 1300 } 1301 // ------------------------------------------------------------------------- 1302 ::rtl::OUString SAL_CALL OSingleSelectQueryComposer::getGroup( ) throw (RuntimeException) 1303 { 1304 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getGroup" ); 1305 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1306 ::osl::MutexGuard aGuard( m_aMutex ); 1307 return getSQLPart(Group,m_aAdditiveIterator,sal_False); 1308 } 1309 // ----------------------------------------------------------------------------- 1310 ::rtl::OUString OSingleSelectQueryComposer::getHavingClause() throw (RuntimeException) 1311 { 1312 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getHavingClause" ); 1313 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1314 ::osl::MutexGuard aGuard( m_aMutex ); 1315 return getSQLPart(Having,m_aAdditiveIterator,sal_False); 1316 } 1317 // ----------------------------------------------------------------------------- 1318 ::rtl::OUString OSingleSelectQueryComposer::getTableAlias(const Reference< XPropertySet >& column) const 1319 { 1320 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getTableAlias" ); 1321 ::rtl::OUString sReturn; 1322 if(m_pTables && m_pTables->getCount() > 1) 1323 { 1324 ::rtl::OUString aCatalog,aSchema,aTable,aComposedName,aColumnName; 1325 column->getPropertyValue(PROPERTY_CATALOGNAME) >>= aCatalog; 1326 column->getPropertyValue(PROPERTY_SCHEMANAME) >>= aSchema; 1327 column->getPropertyValue(PROPERTY_TABLENAME) >>= aTable; 1328 column->getPropertyValue(PROPERTY_NAME) >>= aColumnName; 1329 1330 Sequence< ::rtl::OUString> aNames(m_pTables->getElementNames()); 1331 const ::rtl::OUString* pBegin = aNames.getConstArray(); 1332 const ::rtl::OUString* pEnd = pBegin + aNames.getLength(); 1333 1334 if(!aTable.getLength()) 1335 { // we don't found a table name, now we must search every table for this column 1336 for(;pBegin != pEnd;++pBegin) 1337 { 1338 Reference<XColumnsSupplier> xColumnsSupp; 1339 m_pTables->getByName(*pBegin) >>= xColumnsSupp; 1340 1341 if(xColumnsSupp.is() && xColumnsSupp->getColumns()->hasByName(aColumnName)) 1342 { 1343 // Reference<XPropertySet> xTableProp(xColumnsSupp,UNO_QUERY); 1344 // xTableProp->getPropertyValue(PROPERTY_CATALOGNAME) >>= aCatalog; 1345 // xTableProp->getPropertyValue(PROPERTY_SCHEMANAME) >>= aSchema; 1346 // xTableProp->getPropertyValue(PROPERTY_NAME) >>= aTable; 1347 aTable = *pBegin; 1348 break; 1349 } 1350 } 1351 } 1352 else 1353 { 1354 aComposedName = ::dbtools::composeTableName( m_xMetaData, aCatalog, aSchema, aTable, sal_False, ::dbtools::eInDataManipulation ); 1355 1356 // first check if this is the table we want to or has it a tablealias 1357 1358 if(!m_pTables->hasByName(aComposedName)) 1359 { 1360 ::comphelper::UStringMixLess aTmp(m_aAdditiveIterator.getTables().key_comp()); 1361 ::comphelper::UStringMixEqual aComp(static_cast< ::comphelper::UStringMixLess*>(&aTmp)->isCaseSensitive()); 1362 for(;pBegin != pEnd;++pBegin) 1363 { 1364 Reference<XPropertySet> xTableProp; 1365 m_pTables->getByName(*pBegin) >>= xTableProp; 1366 OSL_ENSURE(xTableProp.is(),"Table isn't a propertyset!"); 1367 if(xTableProp.is()) 1368 { 1369 ::rtl::OUString aCatalog2,aSchema2,aTable2; 1370 xTableProp->getPropertyValue(PROPERTY_CATALOGNAME) >>= aCatalog2; 1371 xTableProp->getPropertyValue(PROPERTY_SCHEMANAME) >>= aSchema2; 1372 xTableProp->getPropertyValue(PROPERTY_NAME) >>= aTable2; 1373 if(aComp(aCatalog,aCatalog2) && aComp(aSchema,aSchema2) && aComp(aTable,aTable2)) 1374 { 1375 aCatalog = aCatalog2; 1376 aSchema = aSchema2; 1377 aTable = aTable2; 1378 break; 1379 } 1380 } 1381 } 1382 } 1383 } 1384 if(pBegin != pEnd) 1385 { 1386 sReturn = ::dbtools::composeTableName( m_xMetaData, aCatalog, aSchema, aTable, sal_True, ::dbtools::eInDataManipulation ); 1387 sReturn += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(".")); 1388 } 1389 } 1390 return sReturn; 1391 } 1392 // ----------------------------------------------------------------------------- 1393 Reference< XIndexAccess > SAL_CALL OSingleSelectQueryComposer::getParameters( ) throw(RuntimeException) 1394 { 1395 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getParameters" ); 1396 // now set the Parameters 1397 if ( !m_aCurrentColumns[ParameterColumns] ) 1398 { 1399 ::vos::ORef< OSQLColumns> aCols = m_aSqlIterator.getParameters(); 1400 ::std::vector< ::rtl::OUString> aNames; 1401 OSQLColumns::Vector::const_iterator aEnd = aCols->get().end(); 1402 for(OSQLColumns::Vector::const_iterator aIter = aCols->get().begin(); aIter != aEnd;++aIter) 1403 aNames.push_back(getString((*aIter)->getPropertyValue(PROPERTY_NAME))); 1404 m_aCurrentColumns[ParameterColumns] = new OPrivateColumns(aCols,m_xMetaData->supportsMixedCaseQuotedIdentifiers(),*this,m_aMutex,aNames,sal_True); 1405 } 1406 1407 return m_aCurrentColumns[ParameterColumns]; 1408 } 1409 // ----------------------------------------------------------------------------- 1410 void OSingleSelectQueryComposer::clearColumns( const EColumnType _eType ) 1411 { 1412 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::clearColumns" ); 1413 OPrivateColumns* pColumns = m_aCurrentColumns[ _eType ]; 1414 if ( pColumns != NULL ) 1415 { 1416 pColumns->disposing(); 1417 m_aColumnsCollection.push_back( pColumns ); 1418 m_aCurrentColumns[ _eType ] = NULL; 1419 } 1420 } 1421 // ----------------------------------------------------------------------------- 1422 void OSingleSelectQueryComposer::clearCurrentCollections() 1423 { 1424 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::clearCurrentCollections" ); 1425 ::std::vector<OPrivateColumns*>::iterator aIter = m_aCurrentColumns.begin(); 1426 ::std::vector<OPrivateColumns*>::iterator aEnd = m_aCurrentColumns.end(); 1427 for (;aIter != aEnd;++aIter) 1428 { 1429 if ( *aIter ) 1430 { 1431 (*aIter)->disposing(); 1432 m_aColumnsCollection.push_back(*aIter); 1433 *aIter = NULL; 1434 } 1435 } 1436 1437 if(m_pTables) 1438 { 1439 m_pTables->disposing(); 1440 m_aTablesCollection.push_back(m_pTables); 1441 m_pTables = NULL; 1442 } 1443 } 1444 // ----------------------------------------------------------------------------- 1445 Reference< XIndexAccess > OSingleSelectQueryComposer::setCurrentColumns( EColumnType _eType, 1446 const ::vos::ORef< OSQLColumns >& _rCols ) 1447 { 1448 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setCurrentColumns" ); 1449 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1450 1451 ::osl::MutexGuard aGuard( m_aMutex ); 1452 // now set the group columns 1453 if ( !m_aCurrentColumns[_eType] ) 1454 { 1455 ::std::vector< ::rtl::OUString> aNames; 1456 OSQLColumns::Vector::const_iterator aEnd = _rCols->get().end(); 1457 for(OSQLColumns::Vector::const_iterator aIter = _rCols->get().begin(); aIter != aEnd;++aIter) 1458 aNames.push_back(getString((*aIter)->getPropertyValue(PROPERTY_NAME))); 1459 m_aCurrentColumns[_eType] = new OPrivateColumns(_rCols,m_xMetaData->supportsMixedCaseQuotedIdentifiers(),*this,m_aMutex,aNames,sal_True); 1460 } 1461 1462 return m_aCurrentColumns[_eType]; 1463 } 1464 // ----------------------------------------------------------------------------- 1465 Reference< XIndexAccess > SAL_CALL OSingleSelectQueryComposer::getGroupColumns( ) throw(RuntimeException) 1466 { 1467 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getGroupColumns" ); 1468 return setCurrentColumns( GroupByColumns, m_aAdditiveIterator.getGroupColumns() ); 1469 } 1470 // ------------------------------------------------------------------------- 1471 Reference< XIndexAccess > SAL_CALL OSingleSelectQueryComposer::getOrderColumns( ) throw(RuntimeException) 1472 { 1473 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getOrderColumns" ); 1474 return setCurrentColumns( OrderColumns, m_aAdditiveIterator.getOrderColumns() ); 1475 } 1476 // ----------------------------------------------------------------------------- 1477 ::rtl::OUString SAL_CALL OSingleSelectQueryComposer::getQueryWithSubstitution( ) throw (SQLException, RuntimeException) 1478 { 1479 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getQueryWithSubstitution" ); 1480 ::osl::MutexGuard aGuard( m_aMutex ); 1481 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1482 1483 ::rtl::OUString sSqlStatement( getQuery() ); 1484 1485 const OSQLParseNode* pStatementNode = m_aSqlIterator.getParseTree(); 1486 if ( pStatementNode ) 1487 { 1488 SQLException aError; 1489 if ( !pStatementNode->parseNodeToExecutableStatement( sSqlStatement, m_xConnection, m_aSqlParser, &aError ) ) 1490 throw SQLException( aError ); 1491 } 1492 1493 return sSqlStatement; 1494 } 1495 // ----------------------------------------------------------------------------- 1496 ::rtl::OUString OSingleSelectQueryComposer::getStatementPart( TGetParseNode& _aGetFunctor, OSQLParseTreeIterator& _rIterator ) 1497 { 1498 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getStatementPart" ); 1499 ::rtl::OUString sResult; 1500 1501 const OSQLParseNode* pNode = _aGetFunctor( &_rIterator ); 1502 if ( pNode ) 1503 pNode->parseNodeToStr( sResult, m_xConnection ); 1504 1505 return sResult; 1506 } 1507 // ----------------------------------------------------------------------------- 1508 namespace 1509 { 1510 ::rtl::OUString lcl_getDecomposedColumnName( const ::rtl::OUString& rComposedName, const ::rtl::OUString& rQuoteString ) 1511 { 1512 const sal_Int32 nQuoteLength = rQuoteString.getLength(); 1513 ::rtl::OUString sName = rComposedName.trim(); 1514 ::rtl::OUString sColumnName; 1515 sal_Int32 nPos, nRPos = 0; 1516 1517 for (;;) 1518 { 1519 nPos = sName.indexOf( rQuoteString, nRPos ); 1520 if ( nPos >= 0 ) 1521 { 1522 nRPos = sName.indexOf( rQuoteString, nPos + nQuoteLength ); 1523 if ( nRPos > nPos ) 1524 { 1525 if ( nRPos + nQuoteLength < sName.getLength() ) 1526 { 1527 nRPos += nQuoteLength; // -1 + 1 skip dot 1528 } 1529 else 1530 { 1531 sColumnName = sName.copy( nPos + nQuoteLength, nRPos - nPos - nQuoteLength ); 1532 break; 1533 } 1534 } 1535 else 1536 break; 1537 } 1538 else 1539 break; 1540 } 1541 return sColumnName.isEmpty() ? rComposedName : sColumnName; 1542 } 1543 1544 ::rtl::OUString lcl_getCondition(const Sequence< Sequence< PropertyValue > >& filter, 1545 const OPredicateInputController& i_aPredicateInputController, 1546 const Reference< XNameAccess >& i_xSelectColumns, 1547 const ::rtl::OUString& rQuoteString ) 1548 { 1549 ::rtl::OUStringBuffer sRet; 1550 const Sequence< PropertyValue >* pOrIter = filter.getConstArray(); 1551 const Sequence< PropertyValue >* pOrEnd = pOrIter + filter.getLength(); 1552 while ( pOrIter != pOrEnd ) 1553 { 1554 if ( pOrIter->getLength() ) 1555 { 1556 sRet.append(L_BRACKET); 1557 const PropertyValue* pAndIter = pOrIter->getConstArray(); 1558 const PropertyValue* pAndEnd = pAndIter + pOrIter->getLength(); 1559 while ( pAndIter != pAndEnd ) 1560 { 1561 sRet.append(pAndIter->Name); 1562 ::rtl::OUString sValue; 1563 pAndIter->Value >>= sValue; 1564 const ::rtl::OUString sColumnName = lcl_getDecomposedColumnName( pAndIter->Name, rQuoteString ); 1565 if ( i_xSelectColumns.is() && i_xSelectColumns->hasByName(sColumnName) ) 1566 { 1567 Reference<XPropertySet> xColumn(i_xSelectColumns->getByName(sColumnName),UNO_QUERY); 1568 sValue = i_aPredicateInputController.getPredicateValue(sValue,xColumn,sal_True); 1569 } 1570 else 1571 { 1572 sValue = i_aPredicateInputController.getPredicateValue(pAndIter->Name,sValue,sal_True); 1573 } 1574 lcl_addFilterCriteria_throw(pAndIter->Handle,sValue,sRet); 1575 ++pAndIter; 1576 if ( pAndIter != pAndEnd ) 1577 sRet.append(STR_AND); 1578 } 1579 sRet.append(R_BRACKET); 1580 } 1581 ++pOrIter; 1582 if ( pOrIter != pOrEnd && sRet.getLength() ) 1583 sRet.append(STR_OR); 1584 } 1585 return sRet.makeStringAndClear(); 1586 } 1587 } 1588 // ----------------------------------------------------------------------------- 1589 void SAL_CALL OSingleSelectQueryComposer::setStructuredFilter( const Sequence< Sequence< PropertyValue > >& filter ) throw (SQLException, ::com::sun::star::lang::IllegalArgumentException, RuntimeException) 1590 { 1591 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setStructuredFilter" ); 1592 OPredicateInputController aPredicateInput(m_aContext.getLegacyServiceFactory(),m_xConnection); 1593 setFilter(lcl_getCondition(filter,aPredicateInput,getColumns(), m_xMetaData->getIdentifierQuoteString())); 1594 } 1595 // ----------------------------------------------------------------------------- 1596 void SAL_CALL OSingleSelectQueryComposer::setStructuredHavingClause( const Sequence< Sequence< PropertyValue > >& filter ) throw (SQLException, RuntimeException) 1597 { 1598 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setStructuredHavingClause" ); 1599 OPredicateInputController aPredicateInput(m_aContext.getLegacyServiceFactory(),m_xConnection); 1600 setHavingClause(lcl_getCondition(filter,aPredicateInput,getColumns(), m_xMetaData->getIdentifierQuoteString())); 1601 } 1602 // ----------------------------------------------------------------------------- 1603 void OSingleSelectQueryComposer::setConditionByColumn( const Reference< XPropertySet >& column, sal_Bool andCriteria ,::std::mem_fun1_t<bool,OSingleSelectQueryComposer,::rtl::OUString>& _aSetFunctor,sal_Int32 filterOperator) 1604 { 1605 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::setConditionByColumn" ); 1606 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1607 1608 if ( !column.is() 1609 || !column->getPropertySetInfo()->hasPropertyByName(PROPERTY_VALUE) 1610 || !column->getPropertySetInfo()->hasPropertyByName(PROPERTY_NAME) 1611 || !column->getPropertySetInfo()->hasPropertyByName(PROPERTY_TYPE)) 1612 throw SQLException(DBACORE_RESSTRING(RID_STR_COLUMN_NOT_VALID),*this,SQLSTATE_GENERAL,1000,Any() ); 1613 1614 sal_Int32 nType = 0; 1615 column->getPropertyValue(PROPERTY_TYPE) >>= nType; 1616 sal_Int32 nSearchable = dbtools::getSearchColumnFlag(m_xConnection,nType); 1617 if(nSearchable == ColumnSearch::NONE) 1618 throw SQLException(DBACORE_RESSTRING(RID_STR_COLUMN_NOT_SEARCHABLE),*this,SQLSTATE_GENERAL,1000,Any() ); 1619 1620 ::osl::MutexGuard aGuard( m_aMutex ); 1621 1622 ::rtl::OUString aName; 1623 column->getPropertyValue(PROPERTY_NAME) >>= aName; 1624 1625 Any aValue; 1626 column->getPropertyValue(PROPERTY_VALUE) >>= aValue; 1627 1628 ::rtl::OUStringBuffer aSQL; 1629 const ::rtl::OUString aQuote = m_xMetaData->getIdentifierQuoteString(); 1630 getColumns(); 1631 1632 if ( m_aCurrentColumns[SelectColumns] && m_aCurrentColumns[SelectColumns]->hasByName(aName) ) 1633 { 1634 Reference<XPropertySet> xColumn; 1635 m_aCurrentColumns[SelectColumns]->getByName(aName) >>= xColumn; 1636 OSL_ENSURE(xColumn->getPropertySetInfo()->hasPropertyByName(PROPERTY_REALNAME),"Property REALNAME not available!"); 1637 OSL_ENSURE(xColumn->getPropertySetInfo()->hasPropertyByName(PROPERTY_TABLENAME),"Property TABLENAME not available!"); 1638 OSL_ENSURE(xColumn->getPropertySetInfo()->hasPropertyByName(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("AggregateFunction"))),"Property AggregateFunctionnot available!"); 1639 1640 ::rtl::OUString sRealName,sTableName; 1641 xColumn->getPropertyValue(PROPERTY_REALNAME) >>= sRealName; 1642 xColumn->getPropertyValue(PROPERTY_TABLENAME) >>= sTableName; 1643 if(sTableName.indexOf('.',0) != -1) 1644 { 1645 ::rtl::OUString aCatlog,aSchema,aTable; 1646 ::dbtools::qualifiedNameComponents(m_xMetaData,sTableName,aCatlog,aSchema,aTable,::dbtools::eInDataManipulation); 1647 sTableName = ::dbtools::composeTableName( m_xMetaData, aCatlog, aSchema, aTable, sal_True, ::dbtools::eInDataManipulation ); 1648 } 1649 else 1650 sTableName = ::dbtools::quoteName(aQuote,sTableName); 1651 1652 if ( !::comphelper::getBOOL(xColumn->getPropertyValue(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("Function")))) ) 1653 { 1654 aSQL = sTableName; 1655 aSQL.appendAscii( "." ); 1656 aSQL.append( ::dbtools::quoteName( aQuote, sRealName ) ); 1657 } 1658 else 1659 aSQL = sRealName; 1660 1661 } 1662 else 1663 { 1664 aSQL = getTableAlias( column ); 1665 aSQL.append( ::dbtools::quoteName( aQuote, aName ) ); 1666 } 1667 1668 1669 if ( aValue.hasValue() ) 1670 { 1671 if( !m_xTypeConverter.is() ) 1672 m_aContext.createComponent( "com.sun.star.script.Converter", m_xTypeConverter ); 1673 OSL_ENSURE(m_xTypeConverter.is(),"NO typeconverter!"); 1674 1675 if ( nType != DataType::BOOLEAN && DataType::BIT != nType ) 1676 { 1677 ::rtl::OUString sEmpty; 1678 lcl_addFilterCriteria_throw(filterOperator,sEmpty,aSQL); 1679 } 1680 1681 switch(nType) 1682 { 1683 case DataType::VARCHAR: 1684 case DataType::CHAR: 1685 case DataType::LONGVARCHAR: 1686 aSQL.append( DBTypeConversion::toSQLString( nType, aValue, sal_True, m_xTypeConverter ) ); 1687 break; 1688 case DataType::CLOB: 1689 { 1690 Reference< XClob > xClob(aValue,UNO_QUERY); 1691 if ( xClob.is() ) 1692 { 1693 const ::sal_Int64 nLength = xClob->length(); 1694 if ( sal_Int64(nLength + aSQL.getLength() + STR_LIKE.getLength() ) < sal_Int64(SAL_MAX_INT32) ) 1695 { 1696 aSQL.appendAscii("'"); 1697 aSQL.append( xClob->getSubString(1,(sal_Int32)nLength) ); 1698 aSQL.appendAscii("'"); 1699 } 1700 } 1701 else 1702 { 1703 aSQL.append( DBTypeConversion::toSQLString( nType, aValue, sal_True, m_xTypeConverter ) ); 1704 } 1705 } 1706 break; 1707 case DataType::VARBINARY: 1708 case DataType::BINARY: 1709 case DataType::LONGVARBINARY: 1710 { 1711 Sequence<sal_Int8> aSeq; 1712 if(aValue >>= aSeq) 1713 { 1714 if(nSearchable == ColumnSearch::CHAR) 1715 { 1716 aSQL.appendAscii( "\'" ); 1717 } 1718 aSQL.appendAscii( "0x" ); 1719 const sal_Int8* pBegin = aSeq.getConstArray(); 1720 const sal_Int8* pEnd = pBegin + aSeq.getLength(); 1721 for(;pBegin != pEnd;++pBegin) 1722 { 1723 aSQL.append( (sal_Int32)*pBegin, 16 ).getStr(); 1724 } 1725 if(nSearchable == ColumnSearch::CHAR) 1726 aSQL.appendAscii( "\'" ); 1727 } 1728 else 1729 throw SQLException(DBACORE_RESSTRING(RID_STR_NOT_SEQUENCE_INT8),*this,SQLSTATE_GENERAL,1000,Any() ); 1730 } 1731 break; 1732 case DataType::BIT: 1733 case DataType::BOOLEAN: 1734 { 1735 sal_Bool bValue = sal_False; 1736 m_xTypeConverter->convertToSimpleType(aValue, TypeClass_BOOLEAN) >>= bValue; 1737 1738 ::rtl::OUString sColumnExp = aSQL.makeStringAndClear(); 1739 getBoleanComparisonPredicate( sColumnExp, bValue, m_nBoolCompareMode, aSQL ); 1740 } 1741 break; 1742 default: 1743 aSQL.append( DBTypeConversion::toSQLString( nType, aValue, sal_True, m_xTypeConverter ) ); 1744 break; 1745 } 1746 } 1747 else 1748 { 1749 sal_Int32 nFilterOp = filterOperator; 1750 if ( filterOperator != SQLFilterOperator::SQLNULL && filterOperator != SQLFilterOperator::NOT_SQLNULL ) 1751 nFilterOp = SQLFilterOperator::SQLNULL; 1752 ::rtl::OUString sEmpty; 1753 lcl_addFilterCriteria_throw(nFilterOp,sEmpty,aSQL); 1754 } 1755 1756 // filter anhaengen 1757 // select ohne where und order by aufbauen 1758 ::rtl::OUString sFilter = getFilter(); 1759 1760 if ( sFilter.getLength() && aSQL.getLength() ) 1761 { 1762 ::rtl::OUString sTemp(L_BRACKET); 1763 sTemp += sFilter; 1764 sTemp += R_BRACKET; 1765 sTemp += andCriteria ? STR_AND : STR_OR; 1766 sFilter = sTemp; 1767 } 1768 sFilter += aSQL.makeStringAndClear(); 1769 1770 // add the filter and the sort order 1771 _aSetFunctor(this,sFilter); 1772 } 1773 // ----------------------------------------------------------------------------- 1774 Sequence< Sequence< PropertyValue > > OSingleSelectQueryComposer::getStructuredCondition( TGetParseNode& _aGetFunctor ) 1775 { 1776 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getStructuredCondition" ); 1777 ::connectivity::checkDisposed(OSubComponent::rBHelper.bDisposed); 1778 1779 MutexGuard aGuard(m_aMutex); 1780 1781 Sequence< Sequence< PropertyValue > > aFilterSeq; 1782 ::rtl::OUString sFilter = getStatementPart( _aGetFunctor, m_aAdditiveIterator ); 1783 1784 if ( sFilter.getLength() != 0 ) 1785 { 1786 ::rtl::OUString aSql(m_aPureSelectSQL); 1787 // build a temporary parse node 1788 const OSQLParseNode* pTempNode = m_aAdditiveIterator.getParseTree(); 1789 1790 aSql += STR_WHERE; 1791 aSql += sFilter; 1792 1793 ::rtl::OUString aErrorMsg; 1794 ::std::auto_ptr<OSQLParseNode> pSqlParseNode( m_aSqlParser.parseTree(aErrorMsg,aSql)); 1795 if ( pSqlParseNode.get() ) 1796 { 1797 m_aAdditiveIterator.setParseTree(pSqlParseNode.get()); 1798 // normalize the filter 1799 OSQLParseNode* pWhereNode = const_cast<OSQLParseNode*>(m_aAdditiveIterator.getWhereTree()); 1800 1801 OSQLParseNode* pCondition = pWhereNode->getChild(1); 1802 #if OSL_DEBUG_LEVEL > 0 1803 ::rtl::OUString sCondition; 1804 pCondition->parseNodeToStr( sCondition, m_xConnection ); 1805 #endif 1806 OSQLParseNode::negateSearchCondition(pCondition); 1807 1808 pCondition = pWhereNode->getChild(1); 1809 #if OSL_DEBUG_LEVEL > 0 1810 sCondition = ::rtl::OUString(); 1811 pCondition->parseNodeToStr( sCondition, m_xConnection ); 1812 #endif 1813 OSQLParseNode::disjunctiveNormalForm(pCondition); 1814 1815 pCondition = pWhereNode->getChild(1); 1816 #if OSL_DEBUG_LEVEL > 0 1817 sCondition = ::rtl::OUString(); 1818 pCondition->parseNodeToStr( sCondition, m_xConnection ); 1819 #endif 1820 OSQLParseNode::absorptions(pCondition); 1821 1822 pCondition = pWhereNode->getChild(1); 1823 #if OSL_DEBUG_LEVEL > 0 1824 sCondition = ::rtl::OUString(); 1825 pCondition->parseNodeToStr( sCondition, m_xConnection ); 1826 #endif 1827 if ( pCondition ) 1828 { 1829 ::std::vector< ::std::vector < PropertyValue > > aFilters; 1830 Reference< XNumberFormatter > xFormatter; 1831 m_aContext.createComponent( "com.sun.star.util.NumberFormatter", xFormatter ); 1832 xFormatter->attachNumberFormatsSupplier( m_xNumberFormatsSupplier ); 1833 1834 if (setORCriteria(pCondition, m_aAdditiveIterator, aFilters, xFormatter)) 1835 { 1836 aFilterSeq.realloc(aFilters.size()); 1837 Sequence<PropertyValue>* pFilters = aFilterSeq.getArray(); 1838 ::std::vector< ::std::vector < PropertyValue > >::const_iterator aEnd = aFilters.end(); 1839 ::std::vector< ::std::vector < PropertyValue > >::const_iterator i = aFilters.begin(); 1840 for ( ; i != aEnd ; ++i) 1841 { 1842 const ::std::vector < PropertyValue >& rProperties = *i; 1843 pFilters->realloc(rProperties.size()); 1844 PropertyValue* pFilter = pFilters->getArray(); 1845 ::std::vector < PropertyValue >::const_iterator j = rProperties.begin(); 1846 ::std::vector < PropertyValue >::const_iterator aEnd2 = rProperties.end(); 1847 for ( ; j != aEnd2 ; ++j) 1848 { 1849 *pFilter = *j; 1850 ++pFilter; 1851 } 1852 ++pFilters; 1853 } 1854 } 1855 } 1856 // restore 1857 m_aAdditiveIterator.setParseTree(pTempNode); 1858 } 1859 } 1860 return aFilterSeq; 1861 } 1862 // ----------------------------------------------------------------------------- 1863 ::rtl::OUString OSingleSelectQueryComposer::getKeyword( SQLPart _ePart ) const 1864 { 1865 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getKeyword" ); 1866 ::rtl::OUString sKeyword; 1867 switch(_ePart) 1868 { 1869 default: 1870 OSL_ENSURE( 0, "OSingleSelectQueryComposer::getKeyWord: Invalid enum value!" ); 1871 // no break, fallback to WHERE 1872 case Where: 1873 sKeyword = STR_WHERE; 1874 break; 1875 case Group: 1876 sKeyword = STR_GROUP_BY; 1877 break; 1878 case Having: 1879 sKeyword = STR_HAVING; 1880 break; 1881 case Order: 1882 sKeyword = STR_ORDER_BY; 1883 break; 1884 } 1885 return sKeyword; 1886 } 1887 1888 // ----------------------------------------------------------------------------- 1889 ::rtl::OUString OSingleSelectQueryComposer::getSQLPart( SQLPart _ePart, OSQLParseTreeIterator& _rIterator, sal_Bool _bWithKeyword ) 1890 { 1891 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "dbaccess", "Ocke.Janssen@sun.com", "OSingleSelectQueryComposer::getSQLPart" ); 1892 TGetParseNode F_tmp(&OSQLParseTreeIterator::getSimpleWhereTree); 1893 ::rtl::OUString sKeyword( getKeyword( _ePart ) ); 1894 switch(_ePart) 1895 { 1896 case Where: 1897 F_tmp = TGetParseNode(&OSQLParseTreeIterator::getSimpleWhereTree); 1898 break; 1899 case Group: 1900 F_tmp = TGetParseNode (&OSQLParseTreeIterator::getSimpleGroupByTree); 1901 break; 1902 case Having: 1903 F_tmp = TGetParseNode(&OSQLParseTreeIterator::getSimpleHavingTree); 1904 break; 1905 case Order: 1906 F_tmp = TGetParseNode(&OSQLParseTreeIterator::getSimpleOrderTree); 1907 break; 1908 default: 1909 OSL_ENSURE(0,"Invalid enum value!"); 1910 } 1911 1912 ::rtl::OUString sRet = getStatementPart( F_tmp, _rIterator ); 1913 if ( _bWithKeyword && sRet.getLength() ) 1914 sRet = sKeyword + sRet; 1915 return sRet; 1916 } 1917 // ----------------------------------------------------------------------------- 1918