Modified the SQLServerDialect in order to replace the distinct keyword with a groub by clause
This commit is contained in:
parent
55c822c56e
commit
74c9ec6bda
|
@ -38,7 +38,9 @@ import org.hibernate.type.StandardBasicTypes;
|
|||
*/
|
||||
public class SQLServerDialect extends AbstractTransactSQLDialect {
|
||||
private static final String SELECT = "select";
|
||||
private static final String SELECT_DISTINCT = "select distinct";
|
||||
private static final String DISTINCT = "distinct";
|
||||
|
||||
|
||||
public SQLServerDialect() {
|
||||
registerColumnType( Types.VARBINARY, "image" );
|
||||
registerColumnType( Types.VARBINARY, 8000, "varbinary($l)" );
|
||||
|
@ -96,38 +98,54 @@ public class SQLServerDialect extends AbstractTransactSQLDialect {
|
|||
.toString();
|
||||
}
|
||||
|
||||
StringBuilder sb = new StringBuilder( querySqlString.trim() );
|
||||
StringBuilder sb = new StringBuilder( querySqlString.trim().toLowerCase() );
|
||||
|
||||
String querySqlLowered = querySqlString.trim().toLowerCase();
|
||||
int orderByIndex = querySqlLowered.toLowerCase().indexOf( "order by" );
|
||||
String orderby = orderByIndex > 0 ? querySqlString.substring( orderByIndex ) : "ORDER BY CURRENT_TIMESTAMP";
|
||||
int orderByIndex = sb.indexOf( "order by" );
|
||||
CharSequence orderby = orderByIndex > 0 ? sb.subSequence( orderByIndex, sb.length() ) : "ORDER BY CURRENT_TIMESTAMP";
|
||||
|
||||
// Delete the order by clause at the end of the query
|
||||
if ( orderByIndex > 0 ) {
|
||||
sb.delete( orderByIndex, orderByIndex + orderby.length() );
|
||||
}
|
||||
|
||||
// Find the end of the select statement
|
||||
int selectIndex = querySqlLowered.trim().indexOf(SELECT_DISTINCT);
|
||||
if (selectIndex != -1) {
|
||||
selectIndex += SELECT_DISTINCT.length();
|
||||
} else {
|
||||
selectIndex = querySqlLowered.trim().indexOf(SELECT);
|
||||
if (selectIndex != -1) {
|
||||
selectIndex += SELECT.length();
|
||||
}
|
||||
}
|
||||
|
||||
// Isert after the select statement the row_number() function:
|
||||
sb.insert(selectIndex, " ROW_NUMBER() OVER (" + orderby + ") as __hibernate_row_nr__,");
|
||||
|
||||
replaceDistinctWithGroupBy( sb );
|
||||
|
||||
insertRowNumberFunction(sb, orderby);
|
||||
|
||||
//Wrap the query within a with statement:
|
||||
sb.insert(0, "WITH query AS (").append(") SELECT * FROM query ");
|
||||
sb.append("WHERE __hibernate_row_nr__ BETWEEN ").append(offset + 1).append(" AND ").append(limit);
|
||||
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
protected static void replaceDistinctWithGroupBy(StringBuilder sb) {
|
||||
int distinctIndex = sb.indexOf( DISTINCT );
|
||||
if (distinctIndex > 0) {
|
||||
|
||||
sb.delete(distinctIndex, distinctIndex + DISTINCT.length() + 1);
|
||||
sb.append(" group by").append(getSelectFieldsWithoutAs(sb));
|
||||
}
|
||||
}
|
||||
|
||||
protected static CharSequence getSelectFieldsWithoutAs(StringBuilder sql) {
|
||||
String select = sql.substring( sql.indexOf(SELECT) + SELECT.length(), sql.indexOf("from"));
|
||||
|
||||
// Strip the as clauses
|
||||
return stripAsStatement(select);
|
||||
}
|
||||
|
||||
protected static String stripAsStatement(String str) {
|
||||
return str.replaceAll("\\sas[^,]+(,?)", "$1");
|
||||
}
|
||||
|
||||
protected static void insertRowNumberFunction(StringBuilder sb, CharSequence orderby) {
|
||||
// Find the end of the select statement
|
||||
int selectEndIndex = sb.indexOf( SELECT ) + SELECT.length();
|
||||
|
||||
// Isert after the select statement the row_number() function:
|
||||
sb.insert( selectEndIndex, " ROW_NUMBER() OVER (" + orderby + ") as __hibernate_row_nr__," );
|
||||
}
|
||||
|
||||
/**
|
||||
* Use <tt>insert table(...) values(...) select SCOPE_IDENTITY()</tt>
|
||||
|
|
|
@ -0,0 +1,42 @@
|
|||
package org.hibernate.dialect;
|
||||
|
||||
import junit.framework.TestCase;
|
||||
|
||||
/**
|
||||
* Unit test of the behavior of the SQLServerDialect utility methods
|
||||
*
|
||||
* @author Valotasion Yoryos
|
||||
*
|
||||
*/
|
||||
public class SQLServerDialectTestCase extends TestCase {
|
||||
|
||||
public void testStripAsStatement() {
|
||||
String input = "some_field1 as f1, some_fild2 as f2, _field3 as f3 ";
|
||||
|
||||
assertEquals( "some_field1, some_fild2, _field3", SQLServerDialect.stripAsStatement(input) );
|
||||
}
|
||||
|
||||
public void testGetSelectFieldsWithoutAs() {
|
||||
StringBuilder input = new StringBuilder( "select some_field1 as f12, some_fild2 as f879, _field3 as _f24674_3 from...." );
|
||||
String output = SQLServerDialect.getSelectFieldsWithoutAs( input ).toString();
|
||||
|
||||
assertEquals( " some_field1, some_fild2, _field3", output );
|
||||
}
|
||||
|
||||
|
||||
public void testReplaceDistinctWithGroupBy() {
|
||||
StringBuilder input = new StringBuilder( "select distinct f1, f2 as ff, f3 from table where f1 = 5" );
|
||||
SQLServerDialect.replaceDistinctWithGroupBy( input );
|
||||
|
||||
assertEquals( "select f1, f2 as ff, f3 from table where f1 = 5 group by f1, f2, f3 ", input.toString() );
|
||||
}
|
||||
|
||||
|
||||
public void testGetLimitString() {
|
||||
String input = "select distinct f1 as f53245 from table849752 order by f234, f67 desc";
|
||||
|
||||
SQLServerDialect sqlDialect = new SQLServerDialect();
|
||||
|
||||
assertEquals( "with query as (select row_number() over (order by f234, f67 desc) as __hibernate_row_nr__, f1 as f53245 from table849752 group by f1) select * from query where __hibernate_row_nr__ between 11 and 15", sqlDialect.getLimitString(input, 10, 15).toLowerCase() );
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue