Skip to content

Commit 3f8e696

Browse files
committed
HHH-17106 Fix ClassCastException when using length 1 named enum mapping
1 parent 66a723e commit 3f8e696

File tree

5 files changed

+128
-25
lines changed

5 files changed

+128
-25
lines changed

hibernate-core/src/main/java/org/hibernate/boot/model/process/internal/EnumeratedValueResolution.java

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -149,9 +149,9 @@ public static <E extends Enum<E>> EnumeratedValueResolution<E,?> fromName(
149149
}
150150
else if ( style == EnumType.STRING ) {
151151
jdbcType = jdbcTypeRegistry.getDescriptor( jdbcTypeIndicators.getColumnLength() == 1 ? CHAR : VARCHAR );
152-
final JavaType<String> jdbcJavaType = jdbcType.getJdbcRecommendedJavaTypeMapping(
153-
jdbcTypeIndicators.getColumnPrecision(),
154-
jdbcTypeIndicators.getColumnScale(),
152+
final JavaType<Object> jdbcJavaType = jdbcType.getJdbcRecommendedJavaTypeMapping(
153+
(int) jdbcTypeIndicators.getColumnLength(),
154+
null,
155155
typeConfiguration
156156
);
157157
converter = new NamedEnumValueConverter<>( enumJavaType, jdbcType, jdbcJavaType );

hibernate-core/src/main/java/org/hibernate/boot/model/process/internal/InferredBasicValueResolver.java

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -334,7 +334,7 @@ public static <E extends Enum<E>, R> EnumeratedValueResolution<E,R> fromEnum(
334334

335335
if ( enumStyle == EnumType.STRING ) {
336336
//noinspection unchecked
337-
return (EnumeratedValueResolution<E, R>) stringEnumValueResolution(
337+
return (EnumeratedValueResolution<E, R>) namedEnumValueResolution(
338338
enumJavaType,
339339
explicitJavaType,
340340
explicitJdbcType,
@@ -418,7 +418,7 @@ private static <N extends Number> JavaType<N> ordinalJavaType(
418418
}
419419
}
420420

421-
private static <E extends Enum<E>> EnumeratedValueResolution<E,String> stringEnumValueResolution(
421+
private static <E extends Enum<E>> EnumeratedValueResolution<E, Object> namedEnumValueResolution(
422422
EnumJavaType<E> enumJavaType,
423423
BasicJavaType<?> explicitJavaType,
424424
JdbcType explicitJdbcType,
@@ -427,7 +427,7 @@ private static <E extends Enum<E>> EnumeratedValueResolution<E,String> stringEnu
427427
final JdbcType jdbcType = explicitJdbcType == null
428428
? enumJavaType.getRecommendedJdbcType( stdIndicators )
429429
: explicitJdbcType;
430-
final JavaType<String> relationalJtd = stringJavaType( explicitJavaType, stdIndicators, context );
430+
final JavaType<Object> relationalJtd = namedJavaType( explicitJavaType, stdIndicators, context );
431431

432432
return new EnumeratedValueResolution<>(
433433
jdbcType,
@@ -442,7 +442,7 @@ private static JdbcType stringJdbcType(JdbcType explicitJdbcType, JdbcTypeIndica
442442
: relationalJtd.getRecommendedJdbcType( stdIndicators );
443443
}
444444

445-
private static JavaType<String> stringJavaType(
445+
private static JavaType<Object> namedJavaType(
446446
BasicJavaType<?> explicitJavaType,
447447
JdbcTypeIndicators stdIndicators,
448448
MetadataBuildingContext context) {
@@ -454,7 +454,7 @@ private static JavaType<String> stringJavaType(
454454
" should handle `java.lang.String` as its relational type descriptor"
455455
);
456456
}
457-
return (JavaType<String>) explicitJavaType;
457+
return (JavaType<Object>) explicitJavaType;
458458
}
459459
else {
460460
return context.getMetadataCollector().getTypeConfiguration().getJavaTypeRegistry()

hibernate-core/src/main/java/org/hibernate/type/EnumType.java

Lines changed: 12 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -273,19 +273,22 @@ private EnumValueConverter<T,?> interpretParameters(Properties parameters) {
273273
);
274274
}
275275

276-
private JavaType<String> getStringType() {
277-
return typeConfiguration.getJavaTypeRegistry().getDescriptor(String.class);
276+
private JavaType<Object> namedJavaType(JdbcTypeIndicators stdIndicators) {
277+
return typeConfiguration.getJavaTypeRegistry().getDescriptor(
278+
stdIndicators.getColumnLength() == 1 ? Character.class : String.class
279+
);
278280
}
279281

280282
private EnumValueConverter<T,?> getConverter(
281283
EnumJavaType<T> enumJavaType,
282284
EnumType<T>.LocalJdbcTypeIndicators localIndicators,
283285
boolean useNamed) {
284-
if (useNamed) {
286+
if ( useNamed ) {
287+
final JavaType<Object> relationalJavaType = namedJavaType( localIndicators );
285288
return new NamedEnumValueConverter<>(
286289
enumJavaType,
287-
getStringType().getRecommendedJdbcType( localIndicators ),
288-
getStringType()
290+
relationalJavaType.getRecommendedJdbcType( localIndicators ),
291+
relationalJavaType
289292
);
290293
}
291294
else {
@@ -310,11 +313,12 @@ private EnumValueConverter<T,?> getConverterForType(
310313
relationalJavaType
311314
);
312315
}
313-
else if ( isCharacterType(type) ) {
316+
else if ( isCharacterType( type ) ) {
317+
final JavaType<Object> relationalJavaType = namedJavaType( localIndicators );
314318
return new NamedEnumValueConverter<>(
315319
enumJavaType,
316-
getStringType().getRecommendedJdbcType( localIndicators ),
317-
getStringType()
320+
relationalJavaType.getRecommendedJdbcType( localIndicators ),
321+
relationalJavaType
318322
);
319323
}
320324
else {

hibernate-core/src/main/java/org/hibernate/type/descriptor/converter/internal/NamedEnumValueConverter.java

Lines changed: 13 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -24,18 +24,19 @@
2424
*
2525
* @author Steve Ebersole
2626
*/
27-
public class NamedEnumValueConverter<E extends Enum<E>> implements EnumValueConverter<E,String>, Serializable {
27+
public class NamedEnumValueConverter<E extends Enum<E>> implements EnumValueConverter<E, Object>, Serializable {
2828
private final EnumJavaType<E> domainTypeDescriptor;
2929
private final JdbcType jdbcType;
30-
private final JavaType<String> relationalTypeDescriptor;
30+
private final JavaType<Object> relationalTypeDescriptor;
3131

3232
public NamedEnumValueConverter(
3333
EnumJavaType<E> domainTypeDescriptor,
3434
JdbcType jdbcType,
35-
JavaType<String> relationalTypeDescriptor) {
35+
JavaType<?> relationalTypeDescriptor) {
3636
this.domainTypeDescriptor = domainTypeDescriptor;
3737
this.jdbcType = jdbcType;
38-
this.relationalTypeDescriptor = relationalTypeDescriptor;
38+
//noinspection unchecked
39+
this.relationalTypeDescriptor = (JavaType<Object>) relationalTypeDescriptor;
3940
}
4041

4142
@Override
@@ -44,18 +45,21 @@ public EnumJavaType<E> getDomainJavaType() {
4445
}
4546

4647
@Override
47-
public JavaType<String> getRelationalJavaType() {
48+
public JavaType<Object> getRelationalJavaType() {
4849
return relationalTypeDescriptor;
4950
}
5051

5152
@Override
52-
public E toDomainValue(String relationalForm) {
53-
return domainTypeDescriptor.fromName( relationalForm );
53+
public E toDomainValue(Object relationalForm) {
54+
return relationalForm == null
55+
? null
56+
: domainTypeDescriptor.fromName( relationalTypeDescriptor.toString( relationalForm ) );
5457
}
5558

5659
@Override
57-
public String toRelationalValue(E domainForm) {
58-
return domainTypeDescriptor.toName( domainForm );
60+
public Object toRelationalValue(E domainForm) {
61+
final String name = domainTypeDescriptor.toName( domainForm );
62+
return name == null ? null : relationalTypeDescriptor.fromString( name );
5963
}
6064

6165
@Override
Lines changed: 95 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
/*
2+
* Hibernate, Relational Persistence for Idiomatic Java
3+
*
4+
* License: GNU Lesser General Public License (LGPL), version 2.1 or later
5+
* See the lgpl.txt file in the root directory or http://www.gnu.org/licenses/lgpl-2.1.html
6+
*/
7+
package org.hibernate.orm.test.mapping.basic;
8+
9+
import java.util.List;
10+
11+
import org.hibernate.testing.orm.junit.DomainModel;
12+
import org.hibernate.testing.orm.junit.JiraKey;
13+
import org.hibernate.testing.orm.junit.SessionFactory;
14+
import org.hibernate.testing.orm.junit.SessionFactoryScope;
15+
import org.junit.jupiter.api.AfterAll;
16+
import org.junit.jupiter.api.BeforeAll;
17+
import org.junit.jupiter.api.Test;
18+
19+
import jakarta.persistence.Column;
20+
import jakarta.persistence.Entity;
21+
import jakarta.persistence.EnumType;
22+
import jakarta.persistence.Enumerated;
23+
import jakarta.persistence.GeneratedValue;
24+
import jakarta.persistence.Id;
25+
26+
import static org.junit.jupiter.api.Assertions.assertEquals;
27+
28+
@SessionFactory
29+
@DomainModel(annotatedClasses = EnumSingleCharTest.Person.class)
30+
@JiraKey("HHH-17106")
31+
public class EnumSingleCharTest {
32+
@BeforeAll
33+
public void setUp(SessionFactoryScope scope) {
34+
scope.inTransaction( session -> {
35+
session.persist( new Person( 1L, SinglecharEnum.B ) );
36+
session.persist( new Person( 2L, SinglecharEnum.R ) );
37+
} );
38+
}
39+
40+
@AfterAll
41+
public void tearDown(SessionFactoryScope scope) {
42+
scope.inTransaction( session -> session.createMutationQuery( "delete from Person" ).executeUpdate() );
43+
}
44+
45+
@Test
46+
public void testUpdate(SessionFactoryScope scope) {
47+
scope.inTransaction( session -> {
48+
List<Person> resultList = session.createQuery( "from Person order by id", Person.class ).getResultList();
49+
assertEquals( 2, resultList.size() );
50+
assertEquals( SinglecharEnum.B, resultList.get( 0 ).getSingleChar() );
51+
assertEquals( SinglecharEnum.R, resultList.get( 1 ).getSingleChar() );
52+
} );
53+
}
54+
55+
public enum SinglecharEnum {
56+
B("first"),
57+
R("second"),
58+
O("third"),
59+
K("fourth"),
60+
E("fifth"),
61+
N("sixth");
62+
63+
private final String item;
64+
65+
SinglecharEnum(String item) {
66+
this.item = item;
67+
}
68+
69+
public String getItem() {
70+
return item;
71+
}
72+
}
73+
74+
@Entity(name = "Person")
75+
public static class Person {
76+
@Id
77+
private Long id;
78+
79+
@Column(name="SINGLE_CHAR", length = 1)
80+
@Enumerated(EnumType.STRING)
81+
private SinglecharEnum singleChar;
82+
83+
public Person() {
84+
}
85+
86+
public Person(Long id, SinglecharEnum singleChar) {
87+
this.id = id;
88+
this.singleChar = singleChar;
89+
}
90+
91+
public SinglecharEnum getSingleChar() {
92+
return singleChar;
93+
}
94+
}
95+
}

0 commit comments

Comments
 (0)