HHH-12292: nulls are valid values for Objects inside array

This commit is contained in:
Yordan Gigov 2018-02-13 17:39:33 +02:00 committed by Vlad Mihalcea
parent 6a44dea137
commit cd86045598
2 changed files with 205 additions and 0 deletions

View File

@ -113,6 +113,9 @@ public class QueryParameterBindingValidator {
}
return false;
}
else if ( value == null) {
return true;
}
else if ( expectedType.isInstance( value ) ) {
return true;
}

View File

@ -0,0 +1,202 @@
/*
* 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.test.hql;
import java.sql.Array;
import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Types;
import java.util.Arrays;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import org.hibernate.dialect.H2Dialect;
import org.hibernate.jpa.test.BaseEntityManagerFunctionalTestCase;
import org.hibernate.query.NativeQuery;
import org.hibernate.type.AbstractSingleColumnStandardBasicType;
import org.hibernate.type.descriptor.ValueBinder;
import org.hibernate.type.descriptor.ValueExtractor;
import org.hibernate.type.descriptor.WrapperOptions;
import org.hibernate.type.descriptor.java.AbstractTypeDescriptor;
import org.hibernate.type.descriptor.java.JavaTypeDescriptor;
import org.hibernate.type.descriptor.sql.BasicBinder;
import org.hibernate.type.descriptor.sql.BasicExtractor;
import org.hibernate.type.descriptor.sql.SqlTypeDescriptor;
import org.hibernate.testing.RequiresDialect;
import org.hibernate.testing.TestForIssue;
import org.junit.Test;
import static org.hibernate.testing.transaction.TransactionUtil.doInHibernate;
import static org.hibernate.testing.transaction.TransactionUtil.doInJPA;
import static org.junit.Assert.assertEquals;
/**
* @author Vlad Mihalcea
*/
@TestForIssue( jiraKey = "HHH-12292")
@RequiresDialect(H2Dialect.class)
public class QueryParametersValidationArrayTest extends BaseEntityManagerFunctionalTestCase {
@Override
protected Class<?>[] getAnnotatedClasses() {
return new Class[] {Event.class};
}
@Test
public void setParameterWithWrongTypeShouldNotThrowIllegalArgumentException() {
doInJPA(this::entityManagerFactory, entityManager -> {
entityManager.createNativeQuery(
"select id " +
"from Event " +
"where readings = :readings" )
.unwrap( NativeQuery.class )
.setParameter( "readings", new String[]{null, "a"}, StringArrayType.INSTANCE )
.getResultList();
});
}
@Entity(name = "Event")
public static class Event {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;
@Column(columnDefinition = "ARRAY(1)")
private String[] readings;
}
public static class StringArrayType
extends AbstractSingleColumnStandardBasicType<String[]> {
public static final StringArrayType INSTANCE = new StringArrayType ();
public StringArrayType() {
super( StringArraySqlTypeDescriptor.INSTANCE, StringArrayTypeDescriptor.INSTANCE);
}
public String getName() {
return "string-array";
}
@Override
protected boolean registerUnderJavaType() {
return true;
}
}
public static class StringArraySqlTypeDescriptor implements SqlTypeDescriptor {
public static final StringArraySqlTypeDescriptor INSTANCE = new StringArraySqlTypeDescriptor();
@Override
public int getSqlType() {
return Types.ARRAY;
}
@Override
public boolean canBeRemapped() {
return true;
}
@Override
public <X> ValueBinder<X> getBinder(JavaTypeDescriptor<X> javaTypeDescriptor) {
return new BasicBinder<X>( javaTypeDescriptor, this) {
@Override
protected void doBind(PreparedStatement st, X value, int index, WrapperOptions options) throws SQLException {
StringArrayTypeDescriptor arrayTypeDescriptor = (StringArrayTypeDescriptor) javaTypeDescriptor;
st.setArray(index, st.getConnection().createArrayOf(
arrayTypeDescriptor.getSqlArrayType(),
arrayTypeDescriptor.unwrap((String[]) value, Object[].class, options)
));
}
@Override
protected void doBind(CallableStatement st, X value, String name, WrapperOptions options)
throws SQLException {
throw new UnsupportedOperationException("Binding by name is not supported!");
}
};
}
@Override
public <X> ValueExtractor<X> getExtractor(final JavaTypeDescriptor<X> javaTypeDescriptor) {
return new BasicExtractor<X>( javaTypeDescriptor, this) {
@Override
protected X doExtract(ResultSet rs, String name, WrapperOptions options) throws SQLException {
return javaTypeDescriptor.wrap(rs.getArray(name), options);
}
@Override
protected X doExtract(CallableStatement statement, int index, WrapperOptions options) throws SQLException {
return javaTypeDescriptor.wrap(statement.getArray(index), options);
}
@Override
protected X doExtract(CallableStatement statement, String name, WrapperOptions options) throws SQLException {
return javaTypeDescriptor.wrap(statement.getArray(name), options);
}
};
}
}
public static class StringArrayTypeDescriptor
extends AbstractTypeDescriptor<String[]> {
public static final StringArrayTypeDescriptor INSTANCE = new StringArrayTypeDescriptor();
public StringArrayTypeDescriptor() {
super(String[].class);
}
public boolean areEqual(String[] one, String[] another) {
if ( one == another ) {
return true;
}
return !( one == null || another == null ) && Arrays.equals( one, another );
}
public String toString(String[] value) {
return Arrays.deepToString( value);
}
@Override
public String[] fromString(String string) {
return null;
}
@SuppressWarnings({"unchecked"})
@Override
public <X> X unwrap(String[] value, Class<X> type, WrapperOptions options) {
return (X) value;
}
@Override
public <X> String[] wrap(X value, WrapperOptions options) {
if (value instanceof Array ) {
Array array = (Array) value;
try {
return (String[]) array.getArray();
} catch (SQLException e) {
throw new IllegalArgumentException(e);
}
}
return (String[]) value;
}
public String getSqlArrayType() {
return "varchar";
}
}
}