From 3e1380a97df11a7fe75878247790a8df0a565f35 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Tom=C3=A1=C5=A1=20Kraus?= Date: Thu, 5 Dec 2024 13:59:37 +0100 Subject: [PATCH] Sorting not honored with JPA and Oracle (#2313) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Fixes Issue #2301 and #2151 Signed-off-by: Tomáš Kraus --- .../framework/TogglingFastTableCreator.java | 59 +++++++++- .../platform/database/OraclePlatform.java | 14 ++- .../jpa/advanced/AdvancedTableCreator.java | 18 +++- .../jpa/advanced/entities/EntityFloat.java | 102 ++++++++++++++++++ .../main/resources/META-INF/persistence.xml | 4 +- .../jpa/jpql/advanced/AdvancedQueryTest.java | 29 ++++- .../jpql/advanced/EntityFloatPopulator.java | 44 ++++++++ 7 files changed, 260 insertions(+), 10 deletions(-) create mode 100644 jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/entities/EntityFloat.java create mode 100644 jpa/eclipselink.jpa.testapps/jpa.test.jpql/src/test/java/org/eclipse/persistence/testing/tests/jpa/jpql/advanced/EntityFloatPopulator.java diff --git a/foundation/org.eclipse.persistence.core.test.framework/src/main/java/org/eclipse/persistence/testing/framework/TogglingFastTableCreator.java b/foundation/org.eclipse.persistence.core.test.framework/src/main/java/org/eclipse/persistence/testing/framework/TogglingFastTableCreator.java index 3902983d29d..3c904a59b55 100644 --- a/foundation/org.eclipse.persistence.core.test.framework/src/main/java/org/eclipse/persistence/testing/framework/TogglingFastTableCreator.java +++ b/foundation/org.eclipse.persistence.core.test.framework/src/main/java/org/eclipse/persistence/testing/framework/TogglingFastTableCreator.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2022 Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2024 Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2010, 2022 IBM Corporation. * Copyright (c) 2010, 2022 Dies Koper (Fujitsu). * @@ -241,15 +241,17 @@ protected static FieldDefinition createNumericFk( * with given name and size and without any additional constraints. * @param name Column name. * @param size Column numeric type size. + * @param subSize Column numeric type sub size. * @param allowNull Allow {@code null} values for column. * @return Initialized {@link FieldDefinition} instance. */ protected static FieldDefinition createNumericColumn( - final String name, final int size, final boolean allowNull) { + final String name, final int size, final int subSize, final boolean allowNull) { final FieldDefinition field = new FieldDefinition(); field.setName(name); field.setTypeName("NUMERIC"); field.setSize(size); + field.setSubSize(subSize); field.setShouldAllowNull(allowNull); field.setIsPrimaryKey(false); field.setUnique(false); @@ -257,6 +259,19 @@ protected static FieldDefinition createNumericColumn( return field; } + /** + * Helper method to create {@link FieldDefinition} instance for numeric column + * with given name and size and without any additional constraints. + * @param name Column name. + * @param size Column numeric type size. + * @param allowNull Allow {@code null} values for column. + * @return Initialized {@link FieldDefinition} instance. + */ + protected static FieldDefinition createNumericColumn( + final String name, final int size, final boolean allowNull) { + return createNumericColumn(name, size, 0, allowNull); + } + /** * Helper method to create {@link FieldDefinition} instance for numeric column * with given name, size of {@code 15}, with {@code null} value allowed and @@ -269,6 +284,44 @@ protected static FieldDefinition createNumericColumn( return createNumericColumn(name, 15, true); } + /** + * Helper method to create {@link FieldDefinition} instance for numeric column + * with given name to store float type values. + * @param name Column name. + * @param allowNull Allow {@code null} values for column. + * @return Initialized {@link FieldDefinition} instance. + */ + protected static FieldDefinition createFloatColumn( + final String name, final boolean allowNull) { + final FieldDefinition field = new FieldDefinition(); + field.setName(name); + field.setType(Float.class); + field.setShouldAllowNull(allowNull); + field.setIsPrimaryKey(false); + field.setUnique(false); + field.setIsIdentity(false); + return field; + } + + /** + * Helper method to create {@link FieldDefinition} instance for numeric column + * with given name to store double type values. + * @param name Column name. + * @param allowNull Allow {@code null} values for column. + * @return Initialized {@link FieldDefinition} instance. + */ + protected static FieldDefinition createDoubleColumn( + final String name, final boolean allowNull) { + final FieldDefinition field = new FieldDefinition(); + field.setName(name); + field.setType(Double.class); + field.setShouldAllowNull(allowNull); + field.setIsPrimaryKey(false); + field.setUnique(false); + field.setIsIdentity(false); + return field; + } + /** * Helper method to create {@link FieldDefinition} instance for DTYPE * column used for inheritance in model. @@ -317,7 +370,7 @@ protected static FieldDefinition createStringColumn( */ protected static FieldDefinition createStringColumn( final String name) { - return createStringColumn(name, 32, true); + return createStringColumn(name, 255, true); } /** diff --git a/foundation/org.eclipse.persistence.core/src/main/java/org/eclipse/persistence/platform/database/OraclePlatform.java b/foundation/org.eclipse.persistence.core/src/main/java/org/eclipse/persistence/platform/database/OraclePlatform.java index f4199e82ad5..6479dec5738 100644 --- a/foundation/org.eclipse.persistence.core/src/main/java/org/eclipse/persistence/platform/database/OraclePlatform.java +++ b/foundation/org.eclipse.persistence.core/src/main/java/org/eclipse/persistence/platform/database/OraclePlatform.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1998, 2022 Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1998, 2024 Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1998, 2021 IBM Corporation. All rights reserved. * * This program and the accompanying materials are made available under the @@ -1077,8 +1077,16 @@ public void printSQLSelectStatement(DatabaseCall call, ExpressionSQLPrinter prin printer.printString(primaryKeyFields); printer.printString(FROM_ID); printer.printString(queryString); - printer.printString(ORDER_BY_ID); - printer.printString(primaryKeyFields); + if (statement.hasOrderByExpressions()) { + try { + statement.printSQLOrderByClause(printer); + } catch (IOException exception) { + throw ValidationException.fileError(exception); + } + } else { + printer.printString(ORDER_BY_ID); + printer.printString(primaryKeyFields); + } printer.printString(END_FROM_ID); printer.printString(MAX_ROW); printer.printParameter(DatabaseCall.MAXROW_FIELD); diff --git a/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/AdvancedTableCreator.java b/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/AdvancedTableCreator.java index e7aec71acfa..d4dd316211f 100644 --- a/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/AdvancedTableCreator.java +++ b/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/AdvancedTableCreator.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1998, 2022 Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1998, 2024 Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1998, 2022 IBM Corporation. All rights reserved. * * This program and the accompanying materials are made available under the @@ -37,6 +37,7 @@ import org.eclipse.persistence.logging.SessionLogEntry; import org.eclipse.persistence.sessions.DatabaseSession; import org.eclipse.persistence.testing.framework.TogglingFastTableCreator; +import org.eclipse.persistence.testing.models.jpa.advanced.entities.EntityFloat; import org.eclipse.persistence.tools.schemaframework.FieldDefinition; import org.eclipse.persistence.tools.schemaframework.ForeignKeyConstraint; import org.eclipse.persistence.tools.schemaframework.TableDefinition; @@ -130,6 +131,7 @@ public AdvancedTableCreator() { addTableDefinition(buildPlanarbeitsgangTable()); addTableDefinition(buildPlanarbeitsgangHistTable()); addTableDefinition(buildMaterialReignisTable()); + addTableDefinition(buildEntityFloatTable()); } public TableDefinition buildADDRESSTable() { @@ -3639,7 +3641,19 @@ public TableDefinition buildMaterialReignisTable() { return tabledefinition; } - + + // Supported data types according to https://docs.oracle.com/cd/E19501-01/819-3659/gcmaz/ + private static TableDefinition buildEntityFloatTable() { + TableDefinition table = new TableDefinition(); + table.setName(EntityFloat.TABLE_NAME); + table.addField(createNumericPk("ID", 10)); + table.addField(createFloatColumn("HEIGHT", false)); + table.addField(createFloatColumn("LENGTH", false)); + table.addField(createFloatColumn("WIDTH", false)); + table.addField(createStringColumn("DESCRIPTION", 255,false)); + return table; + } + @Override public void replaceTables(DatabaseSession session) { DatabasePlatform dbPlatform = session.getPlatform(); diff --git a/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/entities/EntityFloat.java b/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/entities/EntityFloat.java new file mode 100644 index 00000000000..6409f4629e1 --- /dev/null +++ b/jpa/eclipselink.jpa.testapps/jpa.test.advanced/src/main/java/org/eclipse/persistence/testing/models/jpa/advanced/entities/EntityFloat.java @@ -0,0 +1,102 @@ +/* + * Copyright (c) 2024 Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2024 IBM Corporation. All rights reserved. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License v. 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0, + * or the Eclipse Distribution License v. 1.0 which is available at + * http://www.eclipse.org/org/documents/edl-v10.php. + * + * SPDX-License-Identifier: EPL-2.0 OR BSD-3-Clause + */ +package org.eclipse.persistence.testing.models.jpa.advanced.entities; + +import jakarta.persistence.Column; +import jakarta.persistence.Entity; +import jakarta.persistence.Id; +import jakarta.persistence.Table; + +// Based on reproduction scenario from issue #2301 (https://github.com/eclipse-ee4j/eclipselink/issues/2301) +@Entity +@Table(name = EntityFloat.TABLE_NAME) +public class EntityFloat { + + public static final String TABLE_NAME = "ADV_ENTITY_FLOAT"; + + @Id + @Column(name = "ID") + private int id; + + @Column(name = "HEIGHT") + private float height; + + @Column(name = "LENGTH") + private float length; + + @Column(name = "WIDTH") + private float width; + + @Column(name = "DESCRIPTION") + private String description; + + public EntityFloat() { + this(-1, 0f, 0f, 0f, null); + } + + public EntityFloat(int id, float length, float width, float height, String description) { + this.id = id; + this.length = length; + this.width = width; + this.height = height; + this.description = description; + } + + public int getId() { + return id; + } + + public void setId(int id) { + this.id = id; + } + + public float getHeight() { + return height; + } + + public void setHeight(float height) { + this.height = height; + } + + public float getLength() { + return length; + } + + public void setLength(float length) { + this.length = length; + } + + public float getWidth() { + return width; + } + + public void setWidth(float width) { + this.width = width; + } + + public String getDescription() { + return description; + } + + public void setDescription(String description) { + this.description = description; + } + + @Override + public String toString() { + return String.format( + "EntityFloat{ id=%d, length=%f, width=%f, height=%f, description=\"%s\"}", + id, length, width, height, description); + } + +} diff --git a/jpa/eclipselink.jpa.testapps/jpa.test.jpql/src/main/resources/META-INF/persistence.xml b/jpa/eclipselink.jpa.testapps/jpa.test.jpql/src/main/resources/META-INF/persistence.xml index 28ad1e40da0..f458230da04 100644 --- a/jpa/eclipselink.jpa.testapps/jpa.test.jpql/src/main/resources/META-INF/persistence.xml +++ b/jpa/eclipselink.jpa.testapps/jpa.test.jpql/src/main/resources/META-INF/persistence.xml @@ -1,6 +1,6 @@