我面临着一些我无法找到解决方案的问题(或者我正在以错误的方式解决问题)。当我的服务生成某些异常时,似乎序列化失败。这导致获得一个包装异常,这是一个痛苦,因为调用客户端收到一个很差的错误,我想我会从原始异常中丢失有用的信息。
在下面的示例中,它是一个S3存储桶访问问题,但我遇到了一些SQL异常的类似问题。
如果我可以保持我的服务直接干预将是有用的。理想的是添加一些自定义异常序列化代码。
> com.amazonaws.services.simpleworkflow.flow.DataConverterException:
> Failure serializing
> "com.sun.xml.internal.ws.fault.ServerSOAPFaultException: Client
> received SOAP Fault from server: Access Denied (Service: Amazon S3;
> Status Code: 403; Error Code: AccessDenied; Request ID: AAAAAAAAAAAA)
> Please see the server log to find more detail regarding exact cause of
> the failure." of type "class
> com.sun.xml.internal.ws.fault.ServerSOAPFaultException" when mapping
> key "null" at
> com.amazonaws.services.simpleworkflow.flow.JsonDataConverter.throwDataConverterException(JsonDataConverter.java:90)
> at
> com.amazonaws.services.simpleworkflow.flow.JsonDataConverter.toData(JsonDataConverter.java:78)
> at
> com.amazonaws.services.simpleworkflow.flow.pojo.POJOActivityImplementation.throwActivityFailureException(POJOActivityImplementation.java:102)
> at
> com.amazonaws.services.simpleworkflow.flow.pojo.POJOActivityImplementation.execute(POJOActivityImplementation.java:67)
> at
> com.amazonaws.services.simpleworkflow.flow.generic.ActivityImplementationBase.execute(ActivityImplementationBase.java:46)
> at
> com.amazonaws.services.simpleworkflow.flow.worker.SynchronousActivityTaskPoller.execute(SynchronousActivityTaskPoller.java:196)
> at
> com.amazonaws.services.simpleworkflow.flow.worker.ActivityTaskPoller$2.run(ActivityTaskPoller.java:92)
> at
> java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
> at
> java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
> at java.lang.Thread.run(Thread.java:748) Caused by:
> com.fasterxml.jackson.databind.JsonMappingException: Type id handling
> not implemented for type org.w3c.dom.Node (by serializer of type
> com.fasterxml.jackson.databind.ext.DOMSerializer) (through reference
> chain:
> com.sun.xml.internal.ws.fault.ServerSOAPFaultException["fault"]) at
> com.fasterxml.jackson.databind.SerializerProvider.mappingException(SerializerProvider.java:1084)
> at
> com.fasterxml.jackson.databind.JsonSerializer.serializeWithType(JsonSerializer.java:159)
> at
> com.fasterxml.jackson.databind.ser.BeanPropertyWriter.serializeAsField(BeanPropertyWriter.java:695)
> at
> com.fasterxml.jackson.databind.ser.std.BeanSerializerBase.serializeFields(BeanSerializerBase.java:675)
> at
> com.fasterxml.jackson.databind.ser.std.BeanSerializerBase.serializeWithType(BeanSerializerBase.java:566)
> at
> com.fasterxml.jackson.databind.ser.impl.TypeWrappedSerializer.serialize(TypeWrappedSerializer.java:32)
> at
> com.fasterxml.jackson.databind.ser.DefaultSerializerProvider.serializeValue(DefaultSerializerProvider.java:130)
> at
> com.fasterxml.jackson.databind.ObjectMapper._configAndWriteValue(ObjectMapper.java:3559)
> at
> com.fasterxml.jackson.databind.ObjectMapper.writeValueAsString(ObjectMapper.java:2927)
> at
> com.amazonaws.services.simpleworkflow.flow.JsonDataConverter.toData(JsonDataConverter.java:72)
> ... 8 more
感谢您的任何建议
答案 0 :(得分:1)
杰克逊试图序列化一些com.sun.xml.internal.ws.fault.ServerSOAPFaultException
的实例。此异常是javax.xml.ws.soap.SOAPFaultException
的子类,happens to have a getter for retrieving the SOAPFault
。由于SOAPFault
是一个DOM Node
,杰克逊决定使用DOMSerializer
并对代码类型信息进行编码,以便在反序列化时可以找出哪个具体类型fault
是的例子。问题是DOMSerializer
不支持
这种类型的信息,所以你得到com.fasterxml.jackson.databind.JsonMappingException
。
我认为解决此问题的最佳方法是为SOAPFault
注册自定义序列化器/解串器对。然后,您可以序列化为String
并通过SOAPFactory.createFault(...)
methods之一反序列化创建新实例。不要忘记实施JsonSerializer.serializeWithType(...)
,以便杰克逊可以正确地进行打字处理。
(非常粗略)的草案将是
public class SOAPFaultSerializer extends StdSerializer<SOAPFault> {
public SOAPFaultSerializer() {
this(null);
}
public SOAPFaultSerializer(Class<SOAPFault> t) {
super(t);
}
@Override
public void serialize(SOAPFault fault, JsonGenerator jgen, SerializerProvider provider) throws IOException, JsonProcessingException {
jgen.writeStartObject();
// serialize "interesting" SOAPFault information
jgen.writeStringField("faultActor", fault.getFaultActor());
jgen.writeStringField("faultCode", fault.getFaultCode());
...
jgen.writeEndObject();
}
@Override
public void serializeWithType(JsonGenerator jgen, SerializerProvider provider, TypeSerializer typeSer) throws IOException, JsonGenerationException {
typeSer.writeTypePrefixForObject(this, jgen, SOAPFault.class);
serialize(value, jgen, provider);
typeSer.writeTypeSuffixForObject(this, jgen);
}
}
public class SOAPFaultDeserializer extends StdDeserializer<SOAPFault> {
public SOAPFaultDeserializer() {
this(null);
}
public SOAPFaultDeserializer(Class<?> vc) {
super(vc);
}
@Override
public SOAPFault deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException, JsonProcessingException {
JsonNode node = jp.getCodec().readTree(jp);
// deserialize "interesting" SOAPFault information
String faultActor = node.get("faultActor").asText();
String faultCode = node.get("faultCode").asText();
...
SOAPFactory factory = SOAPFactory.newInstance();
SOAPFault fault = factory.createFault();
// fill in SOAPFault with deserialized fields
fault.setFaultActor(faultActor);
fault.setFaultCode(faultCode);
...
return fault;
}
}