Обработка ошибок десериализации Джексона - PullRequest
17 голосов
/ 31 января 2012

Моя проблема довольно проста: у меня есть следующий простой класс:

public class Foo {
   private int id = -1;
   public void setId(int _id){ this.id = _id; }
   public int getId(){ return this.id; }
}

И я пытаюсь обработать следующий JSON:

{
  "id": "blah"
}

Очевидно, здесь есть проблема(«бла» не может быть проанализировано в int)

Раньше Джексон выбрасывал что-то вроде org.codehaus.jackson.map.JsonMappingException: не может создать экземпляр java.lang.Integer из строкового значения «бла»:недопустимое целочисленное значение

Я согласен с этим, но я бы хотел где-нибудь зарегистрировать что-то, позволяющее игнорировать ошибки отображения этого типа.Я пытался с зарегистрированной DeserializationProblemHandler (см. здесь ), но, похоже, он работает только с неизвестными свойствами, а не с проблемами десериализации.

У вас есть какие-либо подсказки по этому вопросу?

Ответы [ 3 ]

14 голосов
/ 01 февраля 2012

Мне удалось решить мою проблему, благодаря Tatu от Jackson ML .

Мне пришлось использовать нестандартные десериализаторы без блокировки для всех примитивных типов, обрабатываемых в Джексоне.Что-то вроде этой фабрики:

public class JacksonNonBlockingObjectMapperFactory {

    /**
     * Deserializer that won't block if value parsing doesn't match with target type
     * @param <T> Handled type
     */
    private static class NonBlockingDeserializer<T> extends JsonDeserializer<T> {
        private StdDeserializer<T> delegate;

        public NonBlockingDeserializer(StdDeserializer<T> _delegate){
            this.delegate = _delegate;
        }

        @Override
        public T deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException, JsonProcessingException {
            try {
                return delegate.deserialize(jp, ctxt);
            }catch (JsonMappingException e){
                // If a JSON Mapping occurs, simply returning null instead of blocking things
                return null;
            }
        }
    }

    private List<StdDeserializer> jsonDeserializers = new ArrayList<StdDeserializer>();

    public ObjectMapper createObjectMapper(){
        ObjectMapper objectMapper = new ObjectMapper();

        SimpleModule customJacksonModule = new SimpleModule("customJacksonModule", new Version(1, 0, 0, null));
        for(StdDeserializer jsonDeserializer : jsonDeserializers){
            // Wrapping given deserializers with NonBlockingDeserializer
            customJacksonModule.addDeserializer(jsonDeserializer.getValueClass(), new NonBlockingDeserializer(jsonDeserializer));
        }

        objectMapper.registerModule(customJacksonModule);
        return objectMapper;
    }

    public JacksonNonBlockingObjectMapperFactory setJsonDeserializers(List<StdDeserializer> _jsonDeserializers){
        this.jsonDeserializers = _jsonDeserializers;
        return this;
    }
}

Затем назовите это следующим образом (передавайте в качестве десериализаторов только те, которые вы не хотите блокировать):

JacksonNonBlockingObjectMapperFactory factory = new JacksonNonBlockingObjectMapperFactory();
factory.setJsonDeserializers(Arrays.asList(new StdDeserializer[]{
    // StdDeserializer, here, comes from Jackson (org.codehaus.jackson.map.deser.StdDeserializer)
    new StdDeserializer.ShortDeserializer(Short.class, null),
    new StdDeserializer.IntegerDeserializer(Integer.class, null),
    new StdDeserializer.CharacterDeserializer(Character.class, null),
    new StdDeserializer.LongDeserializer(Long.class, null),
    new StdDeserializer.FloatDeserializer(Float.class, null),
    new StdDeserializer.DoubleDeserializer(Double.class, null),
    new StdDeserializer.NumberDeserializer(),
    new StdDeserializer.BigDecimalDeserializer(),
    new StdDeserializer.BigIntegerDeserializer(),
    new StdDeserializer.CalendarDeserializer()
}));
ObjectMapper om = factory.createObjectMapper();
8 голосов
/ 09 июля 2014

Возможно, вы захотите, чтобы ваш контроллер справился с проблемой, добавив метод, который обрабатывает это конкретное исключение

@ExceptionHandler(HttpMessageNotReadableException.class)
@ResponseBody
public String handleHttpMessageNotReadableException(HttpMessageNotReadableException ex)
{
    JsonMappingException jme = (JsonMappingException) ex.getCause();
    return jme.getPath().get(0).getFieldName() + " invalid";
}

Конечно, строка

    JsonMappingException jme = (JsonMappingException) ex.getCause();

может бросить приведение классаисключение для некоторых случаев, но я еще не сталкивался с ними.

1 голос
/ 01 декабря 2017

Создание простого Mapper:

@Provider
@Produces(MediaType.APPLICATION_JSON)
public class JSONProcessingErroMapper implements ExceptionMapper<InvalidFormatException> {

@Override
public Response toResponse(InvalidFormatException ex) { 
    return Response.status(400)
             .entity(new ClientError("[User friendly message]"))
             .type(MediaType.APPLICATION_JSON)
             .build();
}

}

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...