Add filter clause for aggregate functions in HQL
Signed-off-by: Jan Schatteman <jschatte@redhat.com>
This commit is contained in:
parent
6447ca9b26
commit
78209dc506
|
@ -182,6 +182,7 @@ EXISTS : [eE] [xX] [iI] [sS] [tT] [sS];
|
||||||
EXP : [eE] [xX] [pP];
|
EXP : [eE] [xX] [pP];
|
||||||
EXTRACT : [eE] [xX] [tT] [rR] [aA] [cC] [tT];
|
EXTRACT : [eE] [xX] [tT] [rR] [aA] [cC] [tT];
|
||||||
FETCH : [fF] [eE] [tT] [cC] [hH];
|
FETCH : [fF] [eE] [tT] [cC] [hH];
|
||||||
|
FILTER : [fF] [iI] [lL] [tT] [eE] [rR];
|
||||||
FIRST : [fF] [iI] [rR] [sS] [tT];
|
FIRST : [fF] [iI] [rR] [sS] [tT];
|
||||||
FLOOR : [fF] [lL] [oO] [oO] [rR];
|
FLOOR : [fF] [lL] [oO] [oO] [rR];
|
||||||
FROM : [fF] [rR] [oO] [mM];
|
FROM : [fF] [rR] [oO] [mM];
|
||||||
|
|
|
@ -690,31 +690,35 @@ aggregateFunction
|
||||||
;
|
;
|
||||||
|
|
||||||
avgFunction
|
avgFunction
|
||||||
: AVG LEFT_PAREN DISTINCT? expression RIGHT_PAREN
|
: AVG LEFT_PAREN DISTINCT? expression RIGHT_PAREN filterClause?
|
||||||
;
|
;
|
||||||
|
|
||||||
sumFunction
|
sumFunction
|
||||||
: SUM LEFT_PAREN DISTINCT? expression RIGHT_PAREN
|
: SUM LEFT_PAREN DISTINCT? expression RIGHT_PAREN filterClause?
|
||||||
;
|
;
|
||||||
|
|
||||||
minFunction
|
minFunction
|
||||||
: MIN LEFT_PAREN DISTINCT? expression RIGHT_PAREN
|
: MIN LEFT_PAREN DISTINCT? expression RIGHT_PAREN filterClause?
|
||||||
;
|
;
|
||||||
|
|
||||||
maxFunction
|
maxFunction
|
||||||
: MAX LEFT_PAREN DISTINCT? expression RIGHT_PAREN
|
: MAX LEFT_PAREN DISTINCT? expression RIGHT_PAREN filterClause?
|
||||||
;
|
;
|
||||||
|
|
||||||
countFunction
|
countFunction
|
||||||
: COUNT LEFT_PAREN DISTINCT? (expression | ASTERISK) RIGHT_PAREN
|
: COUNT LEFT_PAREN DISTINCT? (expression | ASTERISK) RIGHT_PAREN filterClause?
|
||||||
;
|
;
|
||||||
|
|
||||||
everyFunction
|
everyFunction
|
||||||
: (EVERY|ALL) LEFT_PAREN (predicate | subQuery) RIGHT_PAREN
|
: (EVERY|ALL) LEFT_PAREN (predicate | subQuery) RIGHT_PAREN filterClause?
|
||||||
;
|
;
|
||||||
|
|
||||||
anyFunction
|
anyFunction
|
||||||
: (ANY|SOME) LEFT_PAREN (predicate | subQuery) RIGHT_PAREN
|
: (ANY|SOME) LEFT_PAREN (predicate | subQuery) RIGHT_PAREN filterClause?
|
||||||
|
;
|
||||||
|
|
||||||
|
filterClause
|
||||||
|
: FILTER LEFT_PAREN whereClause RIGHT_PAREN
|
||||||
;
|
;
|
||||||
|
|
||||||
standardFunction
|
standardFunction
|
||||||
|
@ -1102,7 +1106,7 @@ rollup
|
||||||
* The lexer hands us recognized keywords using their specific tokens. This is important
|
* The lexer hands us recognized keywords using their specific tokens. This is important
|
||||||
* for the recognition of sqm structure, especially in terms of performance!
|
* for the recognition of sqm structure, especially in terms of performance!
|
||||||
*
|
*
|
||||||
* However we want to continue to allow users to use mopst keywords as identifiers (e.g., attribute names).
|
* However we want to continue to allow users to use most keywords as identifiers (e.g., attribute names).
|
||||||
* This parser rule helps with that. Here we expect that the caller already understands their
|
* This parser rule helps with that. Here we expect that the caller already understands their
|
||||||
* context enough to know that keywords-as-identifiers are allowed.
|
* context enough to know that keywords-as-identifiers are allowed.
|
||||||
*/
|
*/
|
||||||
|
@ -1149,6 +1153,7 @@ identifier
|
||||||
| EXP
|
| EXP
|
||||||
| EXTRACT
|
| EXTRACT
|
||||||
| FETCH
|
| FETCH
|
||||||
|
| FILTER
|
||||||
| FLOOR
|
| FLOOR
|
||||||
| FROM
|
| FROM
|
||||||
| FOR
|
| FOR
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
|
||||||
|
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
|
||||||
|
*/
|
||||||
|
package org.hibernate.dialect.function;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.hibernate.query.sqm.function.AbstractSqmSelfRenderingFunctionDescriptor;
|
||||||
|
import org.hibernate.query.sqm.produce.function.StandardArgumentsValidators;
|
||||||
|
import org.hibernate.query.sqm.produce.function.StandardFunctionReturnTypeResolvers;
|
||||||
|
import org.hibernate.sql.ast.SqlAstTranslator;
|
||||||
|
import org.hibernate.sql.ast.spi.SqlAppender;
|
||||||
|
import org.hibernate.sql.ast.tree.SqlAstNode;
|
||||||
|
import org.hibernate.sql.ast.tree.predicate.Predicate;
|
||||||
|
import org.hibernate.type.StandardBasicTypes;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Jan Schatteman
|
||||||
|
*/
|
||||||
|
public class CaseWhenEveryAnyEmulation extends AbstractSqmSelfRenderingFunctionDescriptor {
|
||||||
|
|
||||||
|
private final boolean every;
|
||||||
|
|
||||||
|
public CaseWhenEveryAnyEmulation(boolean every) {
|
||||||
|
super(
|
||||||
|
every ? "every" : "any",
|
||||||
|
true,
|
||||||
|
StandardArgumentsValidators.exactly( 1 ),
|
||||||
|
StandardFunctionReturnTypeResolvers.invariant( StandardBasicTypes.BOOLEAN )
|
||||||
|
);
|
||||||
|
this.every = every;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void render(
|
||||||
|
SqlAppender sqlAppender,
|
||||||
|
List<SqlAstNode> sqlAstArguments,
|
||||||
|
Predicate filter,
|
||||||
|
SqlAstTranslator<?> walker) {
|
||||||
|
if ( every ) {
|
||||||
|
sqlAppender.appendSql( "min(case when " );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAppender.appendSql( "max(case when " );
|
||||||
|
}
|
||||||
|
if ( filter != null ) {
|
||||||
|
filter.accept( walker );
|
||||||
|
sqlAppender.appendSql( " then case when " );
|
||||||
|
sqlAstArguments.get( 0 ).accept( walker );
|
||||||
|
sqlAppender.appendSql( " then 1 else 0 end else null end)" );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAstArguments.get( 0 ).accept( walker );
|
||||||
|
sqlAppender.appendSql( " then 1 else 0 end)" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void render(
|
||||||
|
SqlAppender sqlAppender, List<SqlAstNode> sqlAstArguments, SqlAstTranslator<?> walker) {
|
||||||
|
this.render( sqlAppender, sqlAstArguments, null, walker );
|
||||||
|
}
|
||||||
|
}
|
|
@ -647,7 +647,7 @@ public class CommonFunctionFactory {
|
||||||
.register();
|
.register();
|
||||||
|
|
||||||
//MySQL has it but how is that even useful?
|
//MySQL has it but how is that even useful?
|
||||||
// queryEngine.getSqmFunctionRegistry().namedTemplateBuilder( "bit_xor" )
|
// queryEngine.getSqmFunctionRegistry().namedTemplateBuilder( "bit_xor" )
|
||||||
// .setExactArgumentCount( 1 )
|
// .setExactArgumentCount( 1 )
|
||||||
// .register();
|
// .register();
|
||||||
}
|
}
|
||||||
|
@ -689,26 +689,14 @@ public class CommonFunctionFactory {
|
||||||
.register();
|
.register();
|
||||||
queryEngine.getSqmFunctionRegistry().registerAlternateKey( "any", "bool_or" );
|
queryEngine.getSqmFunctionRegistry().registerAlternateKey( "any", "bool_or" );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* These are aggregate functions taking one argument,
|
* These are aggregate functions taking one argument,
|
||||||
* for databases that have to emulate the boolean
|
* for databases that have to emulate the boolean
|
||||||
* aggregation functions using sum() and case.
|
* aggregation functions using sum() and case.
|
||||||
*/
|
*/
|
||||||
public static void everyAny_sumCase(QueryEngine queryEngine) {
|
public static void everyAny_sumCase(QueryEngine queryEngine) {
|
||||||
queryEngine.getSqmFunctionRegistry().patternAggregateDescriptorBuilder( "every",
|
queryEngine.getSqmFunctionRegistry().register( "every", new EveryAnyEmulation( true ) );
|
||||||
"(sum(case when ?1 then 0 else 1 end)=0)" )
|
queryEngine.getSqmFunctionRegistry().register( "any", new EveryAnyEmulation( false ) );
|
||||||
.setExactArgumentCount( 1 )
|
|
||||||
.setInvariantType( StandardBasicTypes.BOOLEAN )
|
|
||||||
.setArgumentListSignature("(predicate)")
|
|
||||||
.register();
|
|
||||||
|
|
||||||
queryEngine.getSqmFunctionRegistry().patternAggregateDescriptorBuilder( "any",
|
|
||||||
"(sum(case when ?1 then 1 else 0 end)>0)" )
|
|
||||||
.setExactArgumentCount( 1 )
|
|
||||||
.setInvariantType( StandardBasicTypes.BOOLEAN )
|
|
||||||
.setArgumentListSignature("(predicate)")
|
|
||||||
.register();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -716,39 +704,18 @@ public class CommonFunctionFactory {
|
||||||
* for SQL Server.
|
* for SQL Server.
|
||||||
*/
|
*/
|
||||||
public static void everyAny_sumIif(QueryEngine queryEngine) {
|
public static void everyAny_sumIif(QueryEngine queryEngine) {
|
||||||
queryEngine.getSqmFunctionRegistry().patternAggregateDescriptorBuilder( "every",
|
queryEngine.getSqmFunctionRegistry().register( "every", new SQLServerEveryAnyEmulation( true ) );
|
||||||
"min(iif(?1,1,0))" )
|
queryEngine.getSqmFunctionRegistry().register( "any", new SQLServerEveryAnyEmulation( false ) );
|
||||||
.setExactArgumentCount( 1 )
|
|
||||||
.setInvariantType( StandardBasicTypes.BOOLEAN )
|
|
||||||
.setArgumentListSignature("(predicate)")
|
|
||||||
.register();
|
|
||||||
|
|
||||||
queryEngine.getSqmFunctionRegistry().patternAggregateDescriptorBuilder( "any",
|
|
||||||
"max(iif(?1,1,0))" )
|
|
||||||
.setExactArgumentCount( 1 )
|
|
||||||
.setInvariantType( StandardBasicTypes.BOOLEAN )
|
|
||||||
.setArgumentListSignature("(predicate)")
|
|
||||||
.register();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* These are aggregate functions taking one argument,
|
* These are aggregate functions taking one argument,
|
||||||
* for Oracle.
|
* for Oracle.
|
||||||
*/
|
*/
|
||||||
public static void everyAny_sumCaseCase(QueryEngine queryEngine) {
|
public static void everyAny_sumCaseCase(QueryEngine queryEngine) {
|
||||||
queryEngine.getSqmFunctionRegistry().patternAggregateDescriptorBuilder( "every",
|
queryEngine.getSqmFunctionRegistry().register( "every", new CaseWhenEveryAnyEmulation( true ) );
|
||||||
"min(case when ?1 then 1 else 0 end)" )
|
queryEngine.getSqmFunctionRegistry().register( "any", new CaseWhenEveryAnyEmulation( false ) );
|
||||||
.setExactArgumentCount( 1 )
|
|
||||||
.setInvariantType( StandardBasicTypes.BOOLEAN )
|
|
||||||
.setArgumentListSignature("(predicate)")
|
|
||||||
.register();
|
|
||||||
|
|
||||||
queryEngine.getSqmFunctionRegistry().patternAggregateDescriptorBuilder( "any",
|
|
||||||
"max(case when ?1 then 1 else 0 end)" )
|
|
||||||
.setExactArgumentCount( 1 )
|
|
||||||
.setInvariantType( StandardBasicTypes.BOOLEAN )
|
|
||||||
.setArgumentListSignature("(predicate)")
|
|
||||||
.register();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void yearMonthDay(QueryEngine queryEngine) {
|
public static void yearMonthDay(QueryEngine queryEngine) {
|
||||||
|
|
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
|
||||||
|
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
|
||||||
|
*/
|
||||||
|
package org.hibernate.dialect.function;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.hibernate.metamodel.model.domain.AllowableFunctionReturnType;
|
||||||
|
import org.hibernate.query.spi.QueryEngine;
|
||||||
|
import org.hibernate.query.sqm.function.AbstractSqmSelfRenderingFunctionDescriptor;
|
||||||
|
import org.hibernate.query.sqm.function.SelfRenderingSqmFunction;
|
||||||
|
import org.hibernate.query.sqm.produce.function.ArgumentsValidator;
|
||||||
|
import org.hibernate.query.sqm.produce.function.FunctionReturnTypeResolver;
|
||||||
|
import org.hibernate.query.sqm.produce.function.StandardArgumentsValidators;
|
||||||
|
import org.hibernate.query.sqm.produce.function.StandardFunctionReturnTypeResolvers;
|
||||||
|
import org.hibernate.query.sqm.tree.SqmTypedNode;
|
||||||
|
import org.hibernate.sql.ast.SqlAstTranslator;
|
||||||
|
import org.hibernate.sql.ast.spi.SqlAppender;
|
||||||
|
import org.hibernate.sql.ast.tree.SqlAstNode;
|
||||||
|
import org.hibernate.sql.ast.tree.predicate.Predicate;
|
||||||
|
import org.hibernate.type.StandardBasicTypes;
|
||||||
|
import org.hibernate.type.spi.TypeConfiguration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Jan Schatteman
|
||||||
|
*/
|
||||||
|
public class EveryAnyEmulation extends AbstractSqmSelfRenderingFunctionDescriptor {
|
||||||
|
|
||||||
|
private final boolean every;
|
||||||
|
|
||||||
|
public EveryAnyEmulation(boolean every) {
|
||||||
|
super(
|
||||||
|
every ? "every" : "any",
|
||||||
|
true,
|
||||||
|
StandardArgumentsValidators.exactly( 1 ),
|
||||||
|
StandardFunctionReturnTypeResolvers.invariant( StandardBasicTypes.BOOLEAN )
|
||||||
|
);
|
||||||
|
this.every = every;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void render(
|
||||||
|
SqlAppender sqlAppender,
|
||||||
|
List<SqlAstNode> sqlAstArguments,
|
||||||
|
Predicate filter,
|
||||||
|
SqlAstTranslator<?> walker) {
|
||||||
|
sqlAppender.appendSql( "(sum(case when " );
|
||||||
|
if ( filter != null ) {
|
||||||
|
filter.accept( walker );
|
||||||
|
sqlAppender.appendSql( " then case when " );
|
||||||
|
sqlAstArguments.get( 0 ).accept( walker );
|
||||||
|
if ( every ) {
|
||||||
|
sqlAppender.appendSql( " then 0 else 1 end else null end)=0)" );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAppender.appendSql( " then 1 else 0 end else null end)>0)" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAstArguments.get( 0 ).accept( walker );
|
||||||
|
if ( every ) {
|
||||||
|
sqlAppender.appendSql( " then 0 else 1 end)=0)" );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAppender.appendSql( " then 1 else 0 end)>0)" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void render(
|
||||||
|
SqlAppender sqlAppender, List<SqlAstNode> sqlAstArguments, SqlAstTranslator<?> walker) {
|
||||||
|
this.render( sqlAppender, sqlAstArguments, null, walker );
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,66 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
|
||||||
|
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
|
||||||
|
*/
|
||||||
|
package org.hibernate.dialect.function;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.hibernate.query.sqm.function.AbstractSqmSelfRenderingFunctionDescriptor;
|
||||||
|
import org.hibernate.query.sqm.produce.function.StandardArgumentsValidators;
|
||||||
|
import org.hibernate.query.sqm.produce.function.StandardFunctionReturnTypeResolvers;
|
||||||
|
import org.hibernate.sql.ast.SqlAstTranslator;
|
||||||
|
import org.hibernate.sql.ast.spi.SqlAppender;
|
||||||
|
import org.hibernate.sql.ast.tree.SqlAstNode;
|
||||||
|
import org.hibernate.sql.ast.tree.predicate.Predicate;
|
||||||
|
import org.hibernate.type.StandardBasicTypes;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Jan Schatteman
|
||||||
|
*/
|
||||||
|
public class SQLServerEveryAnyEmulation extends AbstractSqmSelfRenderingFunctionDescriptor {
|
||||||
|
|
||||||
|
private final boolean every;
|
||||||
|
|
||||||
|
public SQLServerEveryAnyEmulation(boolean every) {
|
||||||
|
super(
|
||||||
|
every ? "every" : "any",
|
||||||
|
true,
|
||||||
|
StandardArgumentsValidators.exactly( 1 ),
|
||||||
|
StandardFunctionReturnTypeResolvers.invariant( StandardBasicTypes.BOOLEAN )
|
||||||
|
);
|
||||||
|
this.every = every;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void render(
|
||||||
|
SqlAppender sqlAppender,
|
||||||
|
List<SqlAstNode> sqlAstArguments,
|
||||||
|
Predicate filter,
|
||||||
|
SqlAstTranslator<?> walker) {
|
||||||
|
if ( every ) {
|
||||||
|
sqlAppender.appendSql( "min(iif(" );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAppender.appendSql( "max(iif(" );
|
||||||
|
}
|
||||||
|
if ( filter != null ) {
|
||||||
|
filter.accept( walker );
|
||||||
|
sqlAppender.appendSql( ",iif(" );
|
||||||
|
sqlAstArguments.get( 0 ).accept( walker );
|
||||||
|
sqlAppender.appendSql( ",1,0),null))" );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sqlAstArguments.get( 0 ).accept( walker );
|
||||||
|
sqlAppender.appendSql( ",1,0))" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void render(
|
||||||
|
SqlAppender sqlAppender, List<SqlAstNode> sqlAstArguments, SqlAstTranslator<?> walker) {
|
||||||
|
this.render( sqlAppender, sqlAstArguments, null, walker );
|
||||||
|
}
|
||||||
|
}
|
|
@ -3548,8 +3548,9 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
public SqmExpression<?> visitMaxFunction(HqlParser.MaxFunctionContext ctx) {
|
public SqmExpression<?> visitMaxFunction(HqlParser.MaxFunctionContext ctx) {
|
||||||
final SqmExpression<?> arg = (SqmExpression<?>) ctx.expression().accept( this );
|
final SqmExpression<?> arg = (SqmExpression<?>) ctx.expression().accept( this );
|
||||||
//ignore DISTINCT
|
//ignore DISTINCT
|
||||||
return getFunctionDescriptor("max").generateSqmExpression(
|
return getFunctionDescriptor("max").generateAggregateSqmExpression(
|
||||||
arg,
|
singletonList( arg ),
|
||||||
|
getFilterExpression( ctx.filterClause() ),
|
||||||
(AllowableFunctionReturnType<?>) arg.getNodeType(),
|
(AllowableFunctionReturnType<?>) arg.getNodeType(),
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
|
@ -3560,8 +3561,9 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
public SqmExpression<?> visitMinFunction(HqlParser.MinFunctionContext ctx) {
|
public SqmExpression<?> visitMinFunction(HqlParser.MinFunctionContext ctx) {
|
||||||
final SqmExpression<?> arg = (SqmExpression<?>) ctx.expression().accept( this );
|
final SqmExpression<?> arg = (SqmExpression<?>) ctx.expression().accept( this );
|
||||||
//ignore DISTINCT
|
//ignore DISTINCT
|
||||||
return getFunctionDescriptor("min").generateSqmExpression(
|
return getFunctionDescriptor("min").generateAggregateSqmExpression(
|
||||||
arg,
|
singletonList( arg ),
|
||||||
|
getFilterExpression( ctx.filterClause() ),
|
||||||
(AllowableFunctionReturnType<?>) arg.getNodeType(),
|
(AllowableFunctionReturnType<?>) arg.getNodeType(),
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
|
@ -3575,8 +3577,9 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
? new SqmDistinct<>(arg, getCreationContext().getNodeBuilder())
|
? new SqmDistinct<>(arg, getCreationContext().getNodeBuilder())
|
||||||
: arg;
|
: arg;
|
||||||
|
|
||||||
return getFunctionDescriptor("sum").generateSqmExpression(
|
return getFunctionDescriptor("sum").generateAggregateSqmExpression(
|
||||||
argument,
|
singletonList( argument ),
|
||||||
|
getFilterExpression(ctx.filterClause()),
|
||||||
null,
|
null,
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
|
@ -3593,8 +3596,13 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
|
|
||||||
final SqmExpression<?> argument = (SqmExpression<?>) ctx.predicate().accept( this );
|
final SqmExpression<?> argument = (SqmExpression<?>) ctx.predicate().accept( this );
|
||||||
|
|
||||||
return getFunctionDescriptor("every").generateSqmExpression(
|
if ( argument instanceof SqmSubQuery<?> && ctx.filterClause() != null ) {
|
||||||
argument,
|
throw new SemanticException( "Quantified expression cannot have a filter clause!" );
|
||||||
|
}
|
||||||
|
|
||||||
|
return getFunctionDescriptor("every").generateAggregateSqmExpression(
|
||||||
|
singletonList( argument ),
|
||||||
|
getFilterExpression( ctx.filterClause() ),
|
||||||
resolveExpressableTypeBasic( Boolean.class ),
|
resolveExpressableTypeBasic( Boolean.class ),
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
|
@ -3611,8 +3619,13 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
|
|
||||||
final SqmExpression<?> argument = (SqmExpression<?>) ctx.predicate().accept( this );
|
final SqmExpression<?> argument = (SqmExpression<?>) ctx.predicate().accept( this );
|
||||||
|
|
||||||
return getFunctionDescriptor("any").generateSqmExpression(
|
if ( argument instanceof SqmSubQuery<?> && ctx.filterClause() != null ) {
|
||||||
argument,
|
throw new SemanticException( "Quantified expression cannot have a filter clause!" );
|
||||||
|
}
|
||||||
|
|
||||||
|
return getFunctionDescriptor("any").generateAggregateSqmExpression(
|
||||||
|
singletonList( argument ),
|
||||||
|
getFilterExpression( ctx.filterClause() ),
|
||||||
resolveExpressableTypeBasic( Boolean.class ),
|
resolveExpressableTypeBasic( Boolean.class ),
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
|
@ -3626,8 +3639,9 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
? new SqmDistinct<>( arg, getCreationContext().getNodeBuilder() )
|
? new SqmDistinct<>( arg, getCreationContext().getNodeBuilder() )
|
||||||
: arg;
|
: arg;
|
||||||
|
|
||||||
return getFunctionDescriptor("avg").generateSqmExpression(
|
return getFunctionDescriptor("avg").generateAggregateSqmExpression(
|
||||||
argument,
|
singletonList( argument ),
|
||||||
|
getFilterExpression( ctx.filterClause() ),
|
||||||
resolveExpressableTypeBasic( Double.class ),
|
resolveExpressableTypeBasic( Double.class ),
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
|
@ -3643,14 +3657,22 @@ public class SemanticQueryBuilder<R> extends HqlParserBaseVisitor<Object> implem
|
||||||
? new SqmDistinct<>( arg, getCreationContext().getNodeBuilder() )
|
? new SqmDistinct<>( arg, getCreationContext().getNodeBuilder() )
|
||||||
: arg;
|
: arg;
|
||||||
|
|
||||||
return getFunctionDescriptor("count").generateSqmExpression(
|
return getFunctionDescriptor("count").generateAggregateSqmExpression(
|
||||||
argument,
|
singletonList( argument ),
|
||||||
|
getFilterExpression( ctx.filterClause() ),
|
||||||
resolveExpressableTypeBasic( Long.class ),
|
resolveExpressableTypeBasic( Long.class ),
|
||||||
creationContext.getQueryEngine(),
|
creationContext.getQueryEngine(),
|
||||||
creationContext.getJpaMetamodel().getTypeConfiguration()
|
creationContext.getJpaMetamodel().getTypeConfiguration()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private SqmPredicate getFilterExpression(HqlParser.FilterClauseContext filterClauseCtx) {
|
||||||
|
if (filterClauseCtx == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return (SqmPredicate) filterClauseCtx.whereClause().predicate().accept( this );
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SqmExpression<?> visitCube(HqlParser.CubeContext ctx) {
|
public SqmExpression<?> visitCube(HqlParser.CubeContext ctx) {
|
||||||
return new SqmSummarization<>(
|
return new SqmSummarization<>(
|
||||||
|
|
|
@ -108,6 +108,24 @@ public abstract class AbstractSqmFunctionDescriptor implements SqmFunctionDescri
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final <T> SelfRenderingSqmFunction<T> generateAggregateSqmExpression(
|
||||||
|
List<SqmTypedNode<?>> arguments,
|
||||||
|
SqmPredicate filter,
|
||||||
|
AllowableFunctionReturnType<T> impliedResultType,
|
||||||
|
QueryEngine queryEngine,
|
||||||
|
TypeConfiguration typeConfiguration) {
|
||||||
|
argumentsValidator.validate( arguments );
|
||||||
|
|
||||||
|
return generateSqmAggregateFunctionExpression(
|
||||||
|
arguments,
|
||||||
|
filter,
|
||||||
|
impliedResultType,
|
||||||
|
queryEngine,
|
||||||
|
typeConfiguration
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return an SQM node or subtree representing an invocation of this function
|
* Return an SQM node or subtree representing an invocation of this function
|
||||||
* with the given arguments. This method may be overridden in the case of
|
* with the given arguments. This method may be overridden in the case of
|
||||||
|
|
|
@ -24,7 +24,7 @@ import java.util.List;
|
||||||
* @author Gavin King
|
* @author Gavin King
|
||||||
*/
|
*/
|
||||||
public abstract class AbstractSqmSelfRenderingFunctionDescriptor
|
public abstract class AbstractSqmSelfRenderingFunctionDescriptor
|
||||||
extends AbstractSqmFunctionDescriptor {
|
extends AbstractSqmFunctionDescriptor implements FunctionRenderingSupport {
|
||||||
|
|
||||||
private final boolean isAggregate;
|
private final boolean isAggregate;
|
||||||
|
|
||||||
|
@ -59,7 +59,7 @@ public abstract class AbstractSqmSelfRenderingFunctionDescriptor
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public <T> SelfRenderingSqmFunction<T> generateAggregateSqmExpression(
|
public <T> SelfRenderingSqmAggregateFunction<T> generateSqmAggregateFunctionExpression(
|
||||||
List<SqmTypedNode<?>> arguments,
|
List<SqmTypedNode<?>> arguments,
|
||||||
SqmPredicate filter,
|
SqmPredicate filter,
|
||||||
AllowableFunctionReturnType<T> impliedResultType,
|
AllowableFunctionReturnType<T> impliedResultType,
|
||||||
|
@ -70,7 +70,7 @@ public abstract class AbstractSqmSelfRenderingFunctionDescriptor
|
||||||
}
|
}
|
||||||
return new SelfRenderingSqmAggregateFunction<>(
|
return new SelfRenderingSqmAggregateFunction<>(
|
||||||
this,
|
this,
|
||||||
this::render,
|
this,
|
||||||
arguments,
|
arguments,
|
||||||
filter,
|
filter,
|
||||||
impliedResultType,
|
impliedResultType,
|
||||||
|
@ -95,5 +95,4 @@ public abstract class AbstractSqmSelfRenderingFunctionDescriptor
|
||||||
SqlAstTranslator<?> walker) {
|
SqlAstTranslator<?> walker) {
|
||||||
render( sqlAppender, sqlAstArguments, walker );
|
render( sqlAppender, sqlAstArguments, walker );
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -115,11 +115,16 @@ public class NamedSqmFunctionDescriptor
|
||||||
if ( !firstPass ) {
|
if ( !firstPass ) {
|
||||||
sqlAppender.appendSql( ", " );
|
sqlAppender.appendSql( ", " );
|
||||||
}
|
}
|
||||||
if ( caseWrapper && !( arg instanceof Distinct ) && !( arg instanceof Star ) ) {
|
if ( caseWrapper && !( arg instanceof Distinct ) ) {
|
||||||
sqlAppender.appendSql( "case when " );
|
sqlAppender.appendSql( "case when " );
|
||||||
filter.accept( translator );
|
filter.accept( translator );
|
||||||
sqlAppender.appendSql( " then " );
|
sqlAppender.appendSql( " then " );
|
||||||
|
if ( ( arg instanceof Star ) ) {
|
||||||
|
sqlAppender.appendSql( "1" );
|
||||||
|
}
|
||||||
|
else {
|
||||||
translator.render( arg, argumentRenderingMode );
|
translator.render( arg, argumentRenderingMode );
|
||||||
|
}
|
||||||
sqlAppender.appendSql( " else null end" );
|
sqlAppender.appendSql( " else null end" );
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
|
@ -0,0 +1,268 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* License: GNU Lesser General Public License (LGPL), version 2.1 or later
|
||||||
|
* See the lgpl.txt file in the root directory or http://www.gnu.org/licenses/lgpl-2.1.html
|
||||||
|
*/
|
||||||
|
package org.hibernate.orm.test.query.hql;
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
import org.hibernate.query.Query;
|
||||||
|
import org.hibernate.query.SemanticException;
|
||||||
|
|
||||||
|
import org.hibernate.testing.orm.domain.StandardDomainModel;
|
||||||
|
import org.hibernate.testing.orm.domain.gambit.EntityOfBasics;
|
||||||
|
import org.hibernate.testing.orm.junit.DomainModel;
|
||||||
|
import org.hibernate.testing.orm.junit.ServiceRegistry;
|
||||||
|
import org.hibernate.testing.orm.junit.SessionFactory;
|
||||||
|
import org.hibernate.testing.orm.junit.SessionFactoryScope;
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.Assertions;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertFalse;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Jan Schatteman
|
||||||
|
*/
|
||||||
|
@ServiceRegistry
|
||||||
|
@DomainModel(standardModels = StandardDomainModel.GAMBIT)
|
||||||
|
@SessionFactory
|
||||||
|
public class AggregateFilterClauseTest {
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
public void prepareData(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
em -> {
|
||||||
|
Date now = new Date();
|
||||||
|
|
||||||
|
EntityOfBasics entity1 = new EntityOfBasics();
|
||||||
|
entity1.setId( 1 );
|
||||||
|
entity1.setTheInt( 5 );
|
||||||
|
entity1.setTheInteger( -1 );
|
||||||
|
entity1.setTheDouble( 5.0 );
|
||||||
|
entity1.setTheDate( now );
|
||||||
|
entity1.setTheBoolean( true );
|
||||||
|
em.persist( entity1 );
|
||||||
|
|
||||||
|
EntityOfBasics entity2 = new EntityOfBasics();
|
||||||
|
entity2.setId( 2 );
|
||||||
|
entity2.setTheInt( 6 );
|
||||||
|
entity2.setTheInteger( -2 );
|
||||||
|
entity2.setTheDouble( 6.0 );
|
||||||
|
entity2.setTheBoolean( true );
|
||||||
|
em.persist( entity2 );
|
||||||
|
|
||||||
|
EntityOfBasics entity3 = new EntityOfBasics();
|
||||||
|
entity3.setId( 3 );
|
||||||
|
entity3.setTheInt( 7 );
|
||||||
|
entity3.setTheInteger( 3 );
|
||||||
|
entity3.setTheDouble( 7.0 );
|
||||||
|
entity3.setTheBoolean( false );
|
||||||
|
entity3.setTheDate( new Date(now.getTime() + 200000L) );
|
||||||
|
em.persist( entity3 );
|
||||||
|
|
||||||
|
EntityOfBasics entity4 = new EntityOfBasics();
|
||||||
|
entity4.setId( 4 );
|
||||||
|
entity4.setTheInt( 13 );
|
||||||
|
entity4.setTheInteger( 4 );
|
||||||
|
entity4.setTheDouble( 13.0 );
|
||||||
|
entity4.setTheBoolean( false );
|
||||||
|
entity4.setTheDate( new Date(now.getTime() + 300000L) );
|
||||||
|
em.persist( entity4 );
|
||||||
|
|
||||||
|
EntityOfBasics entity5 = new EntityOfBasics();
|
||||||
|
entity5.setId( 5 );
|
||||||
|
entity5.setTheInteger( 5 );
|
||||||
|
entity5.setTheDouble( 9.0 );
|
||||||
|
entity5.setTheBoolean( false );
|
||||||
|
em.persist( entity5 );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
public void tearDown(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> session.createQuery( "delete from EntityOfBasics" ).executeUpdate()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSimpleSum(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Long expected = 31L;
|
||||||
|
Query q = session.createQuery( "select sum(eob.theInt) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSumWithFilterClause(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Long expected = 11L;
|
||||||
|
Query q = session.createQuery( "select sum(eob.theInt) filter(where eob.theBoolean = true) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSimpleAvg(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Double expected = 8.0D;
|
||||||
|
Query q = session.createQuery( "select avg(eob.theDouble) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAvgWithFilterClause(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Double expected = 5.5D;
|
||||||
|
Query q = session.createQuery( "select avg(eob.theDouble) filter(where eob.theBoolean = true) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSimpleMin(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Double expected = 5D;
|
||||||
|
Query q = session.createQuery( "select min(eob.theDouble) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMinWithFilterClause(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Double expected = 7D;
|
||||||
|
Query q = session.createQuery( "select min(eob.theDouble) filter(where eob.theBoolean = false) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSimpleMax(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Double expected = 13D;
|
||||||
|
Query q = session.createQuery( "select max(eob.theDouble) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMaxWithFilterClause(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Double expected = 6D;
|
||||||
|
Query q = session.createQuery( "select max(eob.theDouble) filter(where eob.theBoolean = true) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSimpleCount(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Long expected = 5L;
|
||||||
|
Query q = session.createQuery( "select count(*) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
|
||||||
|
expected = 3L;
|
||||||
|
q = session.createQuery( "select count(eob.theDate) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCountWithFilterClause(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Long expected = 3L;
|
||||||
|
Query q = session.createQuery( "select count(*) filter(where eob.theBoolean = false) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
|
||||||
|
expected = 2L;
|
||||||
|
q = session.createQuery( "select count(eob.theDate) filter(where eob.theBoolean = false) from EntityOfBasics eob" );
|
||||||
|
assertEquals( expected, q.getSingleResult(), "expected " + expected + ", got " + q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
// poor test verification, but ok ...
|
||||||
|
public void testSimpleEveryAll(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Query q = session.createQuery( "select every( eob.theInteger > 0 ) from EntityOfBasics eob" );
|
||||||
|
assertFalse( (Boolean) q.getSingleResult() );
|
||||||
|
|
||||||
|
q = session.createQuery( "select any( eob.theInteger < 0 ) from EntityOfBasics eob" );
|
||||||
|
assertTrue( (Boolean) q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
// poor test verification, but ok ...
|
||||||
|
public void testEveryAllWithFilterClause(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Query q = session.createQuery( "select every( eob.theInteger > 0 ) filter ( where eob.theBoolean = false ) from EntityOfBasics eob" );
|
||||||
|
assertTrue( (Boolean) q.getSingleResult() );
|
||||||
|
|
||||||
|
q = session.createQuery( "select any( eob.theInteger < 0 ) filter ( where eob.theBoolean = false ) from EntityOfBasics eob" );
|
||||||
|
assertFalse( (Boolean) q.getSingleResult() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIllegalSubquery(SessionFactoryScope scope) {
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Exception e = Assertions.assertThrows(
|
||||||
|
IllegalArgumentException.class,
|
||||||
|
() -> {
|
||||||
|
session.createQuery( "select every( eob.theInteger > 0 ) filter ( where select 1 ) from EntityOfBasics eob" );
|
||||||
|
}
|
||||||
|
|
||||||
|
);
|
||||||
|
assertEquals( SemanticException.class, e.getCause().getClass() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
scope.inTransaction(
|
||||||
|
session -> {
|
||||||
|
Exception e = Assertions.assertThrows(
|
||||||
|
IllegalArgumentException.class,
|
||||||
|
() -> {
|
||||||
|
session.createQuery( "select any( eob.theInteger > 0 ) filter ( where select 1 ) from EntityOfBasics eob" );
|
||||||
|
}
|
||||||
|
|
||||||
|
);
|
||||||
|
assertEquals( SemanticException.class, e.getCause().getClass() );
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue