Skip to content

Commit 134aa74

Browse files
committed
Fix #1883
1 parent 276c130 commit 134aa74

File tree

7 files changed

+28
-22
lines changed

7 files changed

+28
-22
lines changed

release-notes/VERSION

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ Versions: 3.x (for earlier see VERSION-2.x)
1818
(reported by timo-schmid@github)
1919
#1789: Add `createGenerator` methods in `ObjectMapper`, `ObjectWriter`
2020
#1790: Add `createParser` methods in `ObjectMapper`, `ObjectReader`
21+
#1883: Add "abstract type mapping" for deserialization from `Map<ENUMTYPE,V>`
22+
into `EnumMap` (and `Set<ENUMTYPE>` to `EnumSet<EnumType>`)
2123
#1888: Merge `ResolvableSerializer` into `JsonSerializer`, `ResolvableDeserializer`
2224
into `JsonDeserializer`
2325
#1889: Merge `ContextualSerializer` into `JsonSerializer`, `ContextualDeserializer`

src/main/java/com/fasterxml/jackson/databind/ObjectMapper.java

+3-5
Original file line numberDiff line numberDiff line change
@@ -1361,13 +1361,11 @@ public ObjectMapper enableDefaultTyping(DefaultTyping dti) {
13611361
*/
13621362
public ObjectMapper enableDefaultTyping(DefaultTyping applicability, JsonTypeInfo.As includeAs)
13631363
{
1364-
/* 18-Sep-2014, tatu: Let's add explicit check to ensure no one tries to
1365-
* use "As.EXTERNAL_PROPERTY", since that will not work (with 2.5+)
1366-
*/
1364+
// 18-Sep-2014, tatu: Let's add explicit check to ensure no one tries to
1365+
// use "As.EXTERNAL_PROPERTY", since that will not work (with 2.5+)
13671366
if (includeAs == JsonTypeInfo.As.EXTERNAL_PROPERTY) {
13681367
throw new IllegalArgumentException("Cannot use includeAs of "+includeAs);
13691368
}
1370-
13711369
TypeResolverBuilder<?> typer = new DefaultTypeResolverBuilder(applicability);
13721370
// we'll always use full class name, when using defaulting
13731371
typer = typer.init(JsonTypeInfo.Id.CLASS, null);
@@ -1667,7 +1665,7 @@ public DateFormat getDateFormat() {
16671665
*
16681666
* @param hi Instantiator to use; if null, use the default implementation
16691667
*/
1670-
public Object setHandlerInstantiator(HandlerInstantiator hi)
1668+
public ObjectMapper setHandlerInstantiator(HandlerInstantiator hi)
16711669
{
16721670
_deserializationConfig = _deserializationConfig.with(hi);
16731671
_serializationConfig = _serializationConfig.with(hi);

src/main/java/com/fasterxml/jackson/databind/deser/BasicDeserializerFactory.java

+12
Original file line numberDiff line numberDiff line change
@@ -1206,8 +1206,14 @@ public JsonDeserializer<?> createCollectionDeserializer(DeserializationContext c
12061206
if (deser == null) {
12071207
Class<?> collectionClass = type.getRawClass();
12081208
if (contentDeser == null) { // not defined by annotation
1209+
// [databind#1853]: Map `Set<ENUM>` to `EnumSet<ENUM>`
1210+
if (contentType.isEnumType() && (collectionClass == Set.class)) {
1211+
collectionClass = EnumSet.class;
1212+
type = (CollectionType) config.getTypeFactory().constructSpecializedType(type, collectionClass);
1213+
}
12091214
// One special type: EnumSet:
12101215
if (EnumSet.class.isAssignableFrom(collectionClass)) {
1216+
// 25-Jan-2018, tatu: shouldn't we pass `contentDeser`?
12111217
deser = new EnumSetDeserializer(contentType, null);
12121218
}
12131219
}
@@ -1343,6 +1349,12 @@ public JsonDeserializer<?> createMapDeserializer(DeserializationContext ctxt,
13431349
if (deser == null) {
13441350
// Value handling is identical for all, but EnumMap requires special handling for keys
13451351
Class<?> mapClass = type.getRawClass();
1352+
// [databind#1853]: Map `Map<ENUM,x>` to `EnumMap<ENUM,x>`
1353+
if ((mapClass == Map.class) && keyType.isEnumType()) {
1354+
mapClass = EnumMap.class;
1355+
type = (MapType) config.getTypeFactory().constructSpecializedType(type, mapClass);
1356+
// type = (MapType) config.getTypeFactory().constructMapType(mapClass, keyType, contentType);
1357+
}
13461358
if (EnumMap.class.isAssignableFrom(mapClass)) {
13471359
ValueInstantiator inst;
13481360

src/main/java/com/fasterxml/jackson/databind/deser/std/EnumSetDeserializer.java

-5
Original file line numberDiff line numberDiff line change
@@ -30,8 +30,6 @@ public class EnumSetDeserializer
3030
* Specific override for this instance (from proper, or global per-type overrides)
3131
* to indicate whether single value may be taken to mean an unwrapped one-element array
3232
* or not. If null, left to global defaults.
33-
*
34-
* @since 2.7
3533
*/
3634
protected final Boolean _unwrapSingle;
3735

@@ -55,9 +53,6 @@ public EnumSetDeserializer(JavaType enumType, JsonDeserializer<?> deser)
5553
_unwrapSingle = null;
5654
}
5755

58-
/**
59-
* @since 2.7
60-
*/
6156
@SuppressWarnings("unchecked" )
6257
protected EnumSetDeserializer(EnumSetDeserializer base,
6358
JsonDeserializer<?> deser, Boolean unwrapSingle) {

src/test/java/com/fasterxml/jackson/databind/deser/jdk/EnumDeserializationTest.java

+8-7
Original file line numberDiff line numberDiff line change
@@ -345,20 +345,21 @@ public void testAllowUnknownEnumValuesReadAsNullWithCreatorMethod() throws Excep
345345

346346
public void testAllowUnknownEnumValuesForEnumSets() throws Exception
347347
{
348-
ObjectReader reader = MAPPER.reader(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL);
349-
EnumSet<TestEnum> result = reader.forType(new TypeReference<EnumSet<TestEnum>>() { })
348+
EnumSet<TestEnum> result = MAPPER.reader(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL)
349+
.forType(new TypeReference<EnumSet<TestEnum>>() { })
350350
.readValue("[\"NO-SUCH-VALUE\"]");
351351
assertEquals(0, result.size());
352352
}
353-
353+
354354
public void testAllowUnknownEnumValuesAsMapKeysReadAsNull() throws Exception
355355
{
356-
ObjectReader reader = MAPPER.reader(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL);
357-
ClassWithEnumMapKey result = reader.forType(ClassWithEnumMapKey.class)
356+
ClassWithEnumMapKey result = MAPPER.reader(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL)
357+
.forType(ClassWithEnumMapKey.class)
358358
.readValue("{\"map\":{\"NO-SUCH-VALUE\":\"val\"}}");
359-
assertTrue(result.map.containsKey(null));
359+
// 25-Jan-2018, tatu: as per [databind#1883], we upgrade it to `EnumMap`, which won't accept nulls...
360+
assertEquals(0, result.map.size());
360361
}
361-
362+
362363
public void testDoNotAllowUnknownEnumValuesAsMapKeysWhenReadAsNullDisabled() throws Exception
363364
{
364365
assertFalse(MAPPER.isEnabled(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL));

src/test/java/com/fasterxml/jackson/databind/deser/jdk/EnumMapDeserializationTest.java

+3-4
Original file line numberDiff line numberDiff line change
@@ -169,9 +169,8 @@ public void testUnknownKeyAsNull() throws Exception
169169
.readerFor(new TypeReference<Map<TestEnumWithDefault,String>>() { })
170170
.with(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_AS_NULL)
171171
.readValue("{\"unknown\":\"value\"}");
172-
// 04-Jan-2017, tatu: Not sure if this is weird or not, but since `null`s are typically
173-
// ok for "regular" JDK Maps...
174-
assertEquals(1, value2.size());
175-
assertEquals("value", value2.get(null));
172+
// 25-Jan-2018, tatu: as per [databind#1883], we upgrade it to `EnumMap`, which won't accept nulls...
173+
assertEquals(0, value2.size());
174+
assertEquals(EnumMap.class, value2.getClass());
176175
}
177176
}

src/test/java/com/fasterxml/jackson/databind/ser/TestEnumSerialization.java

-1
Original file line numberDiff line numberDiff line change
@@ -214,7 +214,6 @@ public void testEnumUsingToString() throws Exception
214214
assertEquals("\"c2\"", sw.toString());
215215
}
216216

217-
// Test [JACKSON-214]
218217
public void testSubclassedEnums() throws Exception
219218
{
220219
assertEquals("\"B\"", MAPPER.writeValueAsString(EnumWithSubClass.B));

0 commit comments

Comments
 (0)