Estou tendo dificuldade em obter as informações de tipo para uma propriedade genérica para ser mantida quando eu uso Jackson ObjectMapper ir publicando a carga para um valor string.
O resultado que eu gostaria de conseguir é abaixo.
{"@type":"BlockChainWrapper","@id":1,"payload":{"@type":"TestClassA","@id":2,"helloWorld":"Hello World"},"nonce":255,"signature":"SGVsbG8gV29ybGQ="}
O resultado I na verdade é como abaixo (observando os desaparecidos informações @type para o TestClassA.
{"@type":"BlockChainWrapper","@id":1,"payload":{"@id":2,"helloWorld":"Hello World"},"nonce":255,"signature":"SGVsbG8gV29ybGQ="}
O teste que estou usando é:
class BlockChainWrapperTest extends Specification {
def objectMapper = JacksonConfiguration.createObjectMapper()
def "test Json marshalling"() {
given: "A test payload"
def payload = new BlockChainWrapper<TestClassA>(
payload: new TestClassA(),
nonce: 255,
signature: "Hello World".getBytes(Charset.defaultCharset())
)
when:
//ObjectWriter w = objectMapper.writerFor(new TypeReference<BlockChainWrapper<TestClassA>>() { });
//def result = w.writeValueAsString(payload)
def result = objectMapper.writeValueAsString(payload)
then:
assert result == "{\"@type\":\"BlockChainWrapper\",\"@id\":1,\"payload\":{\"@type\":\"TestClassA\",\"@id\":2,\"helloWorld\":\"Hello World\"},\"nonce\":255,\"signature\":\"SGVsbG8gV29ybGQ=\"}"
}
def "test TestClassA generates correct JSON"() {
given:
def payload = new TestClassA()
when:
def result = objectMapper.writeValueAsString(payload)
then:
assert result == "{\"@type\":\"TestClassA\",\"@id\":1,\"helloWorld\":\"Hello World\"}"
}
}
Existem duas versões do teste "teste JSON Marshalling". Tanto o comentada versão, ea versão produtos descomentada exatamente os mesmos resultados.
O segundo teste valida que TestClassA gera as informações de tipo correto em seu próprio direito.
O problema só se torna um problema quando a classe principal com o tipo genérico é incorporada, subsequentemente perder a informação de tipo, e causando deserialisation para retornar uma LinkedHashMap para o conteúdo de TestClassA vez.
TestClassA se parece com:
@JsonRootName("TestClassA")
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, property = "@type")
@JsonIdentityInfo(generator = ObjectIdGenerators.IntSequenceGenerator.class)
public class TestClassA implements Serializable {
private String helloWorld = "Hello World";
}
Os olhares classe genérica gosto:
@JsonRootName("BlockChainWrapper")
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include= JsonTypeInfo.As.PROPERTY, property = "@type")
@JsonIdentityInfo(generator = ObjectIdGenerators.IntSequenceGenerator.class)
public class BlockChainWrapper<T> implements Serializable {
private T payload;
Para completar a configuração Jackson ObjectMapper é
var mapper = new Jackson2ObjectMapperBuilder()
.createXmlMapper(false)
.modules(new JavaTimeModule(), new Jdk8Module())
.serializationInclusion(JsonInclude.Include.NON_NULL)
.build();
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
mapper.configure(JsonGenerator.Feature.WRITE_BIGDECIMAL_AS_PLAIN, true);
mapper.configure(DeserializationFeature.USE_BIG_DECIMAL_FOR_FLOATS, true);
mapper.configure(DeserializationFeature.READ_DATE_TIMESTAMPS_AS_NANOSECONDS, false);
mapper.configure(DeserializationFeature.ADJUST_DATES_TO_CONTEXT_TIME_ZONE, false);
mapper.configure(SerializationFeature.WRITE_DATE_TIMESTAMPS_AS_NANOSECONDS, false);
mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
Acho que fixa o seu problema. Eu criei um projeto github com a solução. https://github.com/GaetanoPiazzolla/stackoverflow-question-61025761
Em suma, você deve colocar o @JsonTypeInfo anotação também em seu Payload campo:
@JsonTypeInfo(use=JsonTypeInfo.Id.NAME, include=JsonTypeInfo.As.PROPERTY, property="@type")
private T payload;