|
3 | 3 | import java.io.IOException;
|
4 | 4 | import java.util.*;
|
5 | 5 |
|
| 6 | +import com.fasterxml.jackson.annotation.JsonCreator; |
| 7 | +import com.fasterxml.jackson.annotation.JsonProperty; |
6 | 8 | import com.fasterxml.jackson.core.*;
|
7 | 9 | import com.fasterxml.jackson.databind.*;
|
8 |
| -import com.fasterxml.jackson.databind.deser.BeanDeserializer; |
9 |
| -import com.fasterxml.jackson.databind.deser.BeanDeserializerBuilder; |
10 |
| -import com.fasterxml.jackson.databind.deser.BeanDeserializerModifier; |
11 | 10 | import com.fasterxml.jackson.databind.deser.std.StdDeserializer;
|
12 | 11 | import com.fasterxml.jackson.databind.deser.std.StdScalarDeserializer;
|
13 | 12 | import com.fasterxml.jackson.databind.module.SimpleModule;
|
@@ -139,7 +138,92 @@ public void setupModule(SetupContext context) {
|
139 | 138 | context.addBeanDeserializerModifier(new Issue476DeserializerModifier());
|
140 | 139 | }
|
141 | 140 | }
|
142 |
| - |
| 141 | + |
| 142 | + public static class Issue1912Bean { |
| 143 | + public Issue1912SubBean subBean; |
| 144 | + |
| 145 | + @JsonCreator(mode = JsonCreator.Mode.PROPERTIES) // This is need to populate _propertyBasedCreator on BeanDeserializerBase |
| 146 | + public Issue1912Bean(@JsonProperty("subBean") Issue1912SubBean subBean) { |
| 147 | + this.subBean = subBean; |
| 148 | + } |
| 149 | + } |
| 150 | + public static class Issue1912SubBean { |
| 151 | + public String a; |
| 152 | + |
| 153 | + public Issue1912SubBean() { } |
| 154 | + |
| 155 | + public Issue1912SubBean(String a) { |
| 156 | + this.a = a; |
| 157 | + } |
| 158 | + } |
| 159 | + |
| 160 | + public static class Issue1912CustomBeanDeserializer extends JsonDeserializer<Issue1912Bean> { |
| 161 | + private BeanDeserializer defaultDeserializer; |
| 162 | + |
| 163 | + public Issue1912CustomBeanDeserializer(BeanDeserializer defaultDeserializer) { |
| 164 | + this.defaultDeserializer = defaultDeserializer; |
| 165 | + } |
| 166 | + |
| 167 | + @Override |
| 168 | + public Issue1912Bean deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException { |
| 169 | + // this is need on some cases, this populate _propertyBasedCreator |
| 170 | + defaultDeserializer.resolve(ctxt); |
| 171 | + |
| 172 | + p.nextFieldName(); // read subBean |
| 173 | + p.nextToken(); // read start object |
| 174 | + |
| 175 | + Issue1912SubBean subBean = (Issue1912SubBean) defaultDeserializer.findProperty("subBean").deserialize(p, ctxt); |
| 176 | + |
| 177 | + return new Issue1912Bean(subBean); |
| 178 | + } |
| 179 | + } |
| 180 | + |
| 181 | + public static class Issue1912CustomPropertyDeserializer extends JsonDeserializer<Issue1912SubBean> { |
| 182 | + |
| 183 | + @Override |
| 184 | + public Issue1912SubBean deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException { |
| 185 | + p.nextFieldName(); // read "a" |
| 186 | + Issue1912SubBean object = new Issue1912SubBean(p.nextTextValue() + "_custom"); |
| 187 | + p.nextToken(); |
| 188 | + return object; |
| 189 | + } |
| 190 | + } |
| 191 | + public static class Issue1912UseAddOrReplacePropertyDeserializerModifier extends BeanDeserializerModifier { |
| 192 | + |
| 193 | + @Override |
| 194 | + public JsonDeserializer<?> modifyDeserializer(DeserializationConfig config, BeanDescription beanDesc, JsonDeserializer<?> deserializer) { |
| 195 | + if (beanDesc.getBeanClass() == Issue1912Bean.class) { |
| 196 | + return new Issue1912CustomBeanDeserializer((BeanDeserializer) deserializer); |
| 197 | + } |
| 198 | + return super.modifyDeserializer(config, beanDesc, deserializer); |
| 199 | + } |
| 200 | + |
| 201 | + @Override |
| 202 | + public BeanDeserializerBuilder updateBuilder(DeserializationConfig config, BeanDescription beanDesc, BeanDeserializerBuilder builder) { |
| 203 | + if (beanDesc.getBeanClass() == Issue1912Bean.class) { |
| 204 | + Iterator<SettableBeanProperty> props = builder.getProperties(); |
| 205 | + while (props.hasNext()) { |
| 206 | + SettableBeanProperty prop = props.next(); |
| 207 | + SettableBeanProperty propWithCustomDeserializer = prop.withValueDeserializer(new Issue1912CustomPropertyDeserializer()); |
| 208 | + builder.addOrReplaceProperty(propWithCustomDeserializer, true); |
| 209 | + } |
| 210 | + } |
| 211 | + |
| 212 | + return builder; |
| 213 | + } |
| 214 | + } |
| 215 | + public class Issue1912Module extends SimpleModule { |
| 216 | + |
| 217 | + public Issue1912Module() { |
| 218 | + super("Issue1912Module", Version.unknownVersion()); |
| 219 | + } |
| 220 | + |
| 221 | + @Override |
| 222 | + public void setupModule(SetupContext context) { |
| 223 | + context.addBeanDeserializerModifier(new Issue1912UseAddOrReplacePropertyDeserializerModifier()); |
| 224 | + } |
| 225 | + } |
| 226 | + |
143 | 227 | // [Issue#121], arrays, collections, maps
|
144 | 228 |
|
145 | 229 | enum EnumABC { A, B, C; }
|
@@ -380,4 +464,11 @@ public JsonDeserializer<?> modifyDeserializer(DeserializationConfig config,
|
380 | 464 | assertEquals("ABCDEF", result);
|
381 | 465 | }
|
382 | 466 |
|
| 467 | + public void testAddOrReplacePropertyIsUsedOnDeserialization() throws Exception { |
| 468 | + ObjectMapper mapper = new ObjectMapper(); |
| 469 | + mapper.registerModule(new Issue1912Module()); |
| 470 | + |
| 471 | + Issue1912Bean result = mapper.readValue("{\"subBean\": {\"a\":\"foo\"}}", Issue1912Bean.class); |
| 472 | + assertEquals("foo_custom", result.subBean.a); |
| 473 | + } |
383 | 474 | }
|
0 commit comments