From f53a29ab12e0eda878a68c3ae974c6eaf5accfb8 Mon Sep 17 00:00:00 2001 From: Jan Schatteman Date: Thu, 12 Jan 2023 23:20:55 +0100 Subject: [PATCH] HHH-16020 - Fix for incorrect offset parameter index and add test for issue Signed-off-by: Jan Schatteman --- .../pagination/AbstractLimitHandler.java | 2 +- .../query/NativeQueryLimitOffsetTest.java | 132 ++++++++++++++++++ 2 files changed, 133 insertions(+), 1 deletion(-) create mode 100644 hibernate-core/src/test/java/org/hibernate/orm/test/query/NativeQueryLimitOffsetTest.java diff --git a/hibernate-core/src/main/java/org/hibernate/dialect/pagination/AbstractLimitHandler.java b/hibernate-core/src/main/java/org/hibernate/dialect/pagination/AbstractLimitHandler.java index ea924d7dcc..4930146d88 100644 --- a/hibernate-core/src/main/java/org/hibernate/dialect/pagination/AbstractLimitHandler.java +++ b/hibernate-core/src/main/java/org/hibernate/dialect/pagination/AbstractLimitHandler.java @@ -194,7 +194,7 @@ public abstract class AbstractLimitHandler implements LimitHandler { if ( bindOffset ) { statement.setInt( - index + ( reverse || !bindLimit ? 1 : 0 ), + index + ( reverse && bindLimit ? 1 : 0 ), getFirstRow( limit ) ); } diff --git a/hibernate-core/src/test/java/org/hibernate/orm/test/query/NativeQueryLimitOffsetTest.java b/hibernate-core/src/test/java/org/hibernate/orm/test/query/NativeQueryLimitOffsetTest.java new file mode 100644 index 0000000000..f95f0c2c8a --- /dev/null +++ b/hibernate-core/src/test/java/org/hibernate/orm/test/query/NativeQueryLimitOffsetTest.java @@ -0,0 +1,132 @@ +/* + * 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; + +import java.util.List; + +import org.hibernate.dialect.AbstractHANADialect; +import org.hibernate.dialect.MySQLDialect; +import org.hibernate.dialect.OracleDialect; +import org.hibernate.dialect.SpannerDialect; + +import jakarta.persistence.Entity; +import jakarta.persistence.Id; + +import org.hibernate.testing.TestForIssue; +import org.hibernate.testing.orm.junit.DomainModel; +import org.hibernate.testing.orm.junit.SessionFactory; +import org.hibernate.testing.orm.junit.SessionFactoryScope; +import org.hibernate.testing.orm.junit.SkipForDialect; + +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +/** + * @author Jan Schatteman + */ +@DomainModel( + annotatedClasses = { NativeQueryLimitOffsetTest.Person.class } +) +@SessionFactory +public class NativeQueryLimitOffsetTest { + + @BeforeEach + public void setup(SessionFactoryScope scope) { + scope.inTransaction( + session -> { + session.persist( new Person( 1L, "John" ) ); + session.persist( new Person( 2L, "Jack" ) ); + session.persist( new Person( 3L, "Bob" ) ); + session.persist( new Person( 4L, "Jill" ) ); + session.persist( new Person( 5L, "Jane" ) ); + session.persist( new Person( 6L, "Anne" ) ); + session.persist( new Person( 7L, "Joe" ) ); + } + ); + } + + @AfterEach + public void tearDown(SessionFactoryScope scope) { + scope.inTransaction( + session -> { + session.createMutationQuery( "delete from Person" ).executeUpdate(); + } + ); + } + + @Test + @TestForIssue( jiraKey = "HHH-16020" ) + public void testFullLimitOffsetOnNativeQuery(SessionFactoryScope scope) { + scope.inTransaction( + session -> { + List l = session.createNativeQuery( "select id from Person where name like :name", Long.class) + .setParameter("name", "J%") + .setFirstResult( 1 ) + .setMaxResults( 4 ) + .getResultList(); + assertEquals( 2, l.get( 0 ) ); + assertEquals( 4, l.size() ); + } + ); + } + + @Test + @TestForIssue( jiraKey = "HHH-16020" ) + @SkipForDialect( dialectClass = MySQLDialect.class, matchSubTypes = true) + @SkipForDialect( dialectClass = OracleDialect.class, majorVersion = 12, minorVersion = 2, matchSubTypes = true) + @SkipForDialect( dialectClass = AbstractHANADialect.class, matchSubTypes = true) + @SkipForDialect( dialectClass = SpannerDialect.class, matchSubTypes = true) + public void testPartialLimitOffsetOnNativeQuery(SessionFactoryScope scope) { + scope.inTransaction( + session -> { + List l = session.createNativeQuery( "select id from Person where name like :name", Long.class) + .setParameter("name", "J%") + .setFirstResult(1) + .getResultList(); + assertEquals( 2, l.get( 0 ) ); + assertEquals( 4, l.size() ); + + l = session.createNativeQuery( "select id from Person where name like :name", Long.class) + .setParameter("name", "J%") + .setMaxResults( 3 ) + .getResultList(); + assertEquals( 1, l.get( 0 ) ); + assertEquals( 3, l.size() ); + } + ); + } + + @Entity(name = "Person") + public static class Person { + @Id + private Long id; + private String name; + + public Person(Long id, String name) { + this.id = id; + this.name = name; + } + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + } +}