Jackson 的 readEntity 正在将我的异常序列化为 java.lang.Throwable 而不是正确的错误异常



在我的代码中,我有一个扩展IllegalArgumentException的异常:

public class InvalidDataSourceException extends IllegalArgumentException {
public InvalidDataSourceException(String message, Throwable cause) {
super(message, cause);
}
}

在我的泽西岛 REST Web 服务中,所有错误都用类RESTError包装,以便 所有 REST 服务都可以引发相同的异常:

public class RESTError extends Throwable {
public RESTError(Throwable cause) {
super(cause);
}
}

因此,当 InvalidDataSourceException 发生时,它被抛出为RESTError的原因:

throw new RESTError(invalidDataSourceException);

我们已经为 Throwable 对象注册了一个序列化程序:

public class ThrowableSerializer extends JsonSerializer<Throwable> {
@Override
public void serialize(Throwable value, JsonGenerator jgen,
SerializerProvider provider) throws IOException, JsonProcessingException {
jgen.writeStartObject();
jgen.writeStringField("class", value.getClass().getCanonicalName());
jgen.writeStringField("message", value.getMessage());
jgen.writeObjectField("stackTrace", value.getStackTrace());
jgen.writeStringField("stackTraceAsString", Throwables.getStackTraceAsString(value));
jgen.writeObjectField("cause", value.getCause());
jgen.writeEndObject();
}
}

下面是一个单元测试,其中我尝试将 RESTError 写入字符串,然后重新序列化:

public class TestJsonSerializationOfExceptions {
private static final Logger log = LoggerFactory.getLogger(TestJsonSerializationOfExceptions.class);
@Test
public void testBasic() throws Exception {
ObjectMapper objectMapper = ObjectMapperFactory.create();
try {
throw new InvalidDataSourceException("test msg");
} catch (Throwable t) {
RESTError restError = new RESTError(Response.Status.INTERNAL_SERVER_ERROR, Errors.General, t);
String str = objectMapper.writeValueAsString(restError); // It is correct here!
log.info("Rest error when serialized: {}", str);
restError = objectMapper.readValue(str, RESTError.class);
Assert.assertEquals(restError.getCause().getClass(), InvalidDataSourceException.class);
}
}
}

当我这样做时,异常原因被序列化为java.lang.Throwable,而不是我的自定义InvalidDataSourceException。它把它放下了。因此,单元测试失败。为什么?

看看 Throwable javadoc。

当您扩展 Throwable 并将另一个传递给构造函数时,它会传递给private Throwable cause属性。然后这是关于类型的唯一信息。因此,在反序列化期间,Java只知道存在类型Throwable的属性cause,并且不知道它是InvalidDataSourceException还是其他任何信息。

编辑:现在看起来缺少反序列化程序问题

最新更新