JsonProperty 会序列化但不会反序列化

JsonProperty will serialize but not deserialize

简而言之,我正在尝试为我编写的休息服务编写一个放心的集成测试。我的问题是,当服务器尝试反序列化 JSON 时,我收到字段 "meta-data" 未标记为可忽略的错误。我必须处理这个字段,这也意味着(据我所知)我不能将其标记为可忽略。是的,我有 "turned it off and on again"。关闭 Apache,手动删除我的 war 和分解文件,重新部署,检查内容以查看更新,重新打开服务器。

免责声明:如果我不能升级版本,但我可能会降级。例如,我可以使用 Jackson 1.0 而不是 Jackson 2.0。为了简洁起见,我还简化了我的问题。我无法以任何方式更改 JSON 结构。我看过与 Spring 相关的类似主题,但我没有使用 Spring。

这只是整个 JSON 对象的一个​​片段。如果我能让元数据及其两个属性起作用,我就可以将我学到的知识应用到问题的其余部分。谢谢!

我的JSON

{
  "meta-data" : {
    "submission-time" : "06.12.2016",
    "workflow-reference" : "TEST-WORKFLOW"
  }
}

我的JSONPOJO

package com.company.group.project.module.shared.json;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonFormat.Feature;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import java.util.List;

@JsonIgnoreProperties(ignoreUnknown = true)
public class JsonMandateRequest {

    @JsonProperty("meta-data")
    private MetaData metadata;

    public MetaData getMetadata() {
        if (null == metadata) {
            metadata = new MetaData();
        }
        return metadata;
    }

    public void setMetadata(final MetaData metadata) {
        this.metadata = metadata;
    }

    @JsonSerialize
    public static class MetaData {

        @JsonProperty("submission-time")
        private String timestamp;

        @JsonProperty("workflow-reference")
        private String workflow;

        public String getTimestamp() {
            return timestamp;
        }

        public void setTimestamp(final String timestamp) {
            this.timestamp = timestamp;
        }

        public String getWorkflow() {
            return workflow;
        }

        public void setWorkflow(final String workflow) {
            this.workflow = workflow;
        }

    }
}

我的 Rest Service 声明(我知道路径有效。那不是坏的。)

@POST
@Consumes(MediaType.APPLICATION_JSON)
@Produces({MediaType.APPLICATION_JSON, MediaType.TEXT_PLAIN})
public Response uploadFile(final JsonMandateRequest reminders) {
    return buildResponse(reminders, PROCESSOR_UPLOAD_MANDATE);
}

这是生成 JSON 字符串的代码。它有效,但我将其包括在内,以防它以某种方式使 reader 受益。

public static String writeObject(final Object obj) throws IOException
{
    ObjectMapper mapper = new ObjectMapper();
    mapper = mapper.disable(
            DeserializationFeature.FAIL_ON_INVALID_SUBTYPE);
    return mapper.writerWithDefaultPrettyPrinter().writeValueAsString(obj);
}

我的集成测试代码片段

    Response r = given()
            .contentType(ContentType.JSON)
            .body(jsonString)
            .when().post(url);
    System.out.println("Response from " + url + "(" + r.getStatusCode() + ")");
    System.out.println(r.asString());

输出:

Response from http://localhost:8080/application/rest/path(400)
Unrecognized field "meta-data" (Class com.company.group.project.module.shared.json.JsonMandateRequest), not marked as ignorable
 at [Source: org.apache.catalina.connector.CoyoteInputStream@b60721a; line: 2, column: 18] (through reference chain: com.company.group.project.module.shared.json.JsonMandateRequest["meta-data"])
Completed com.company.group.project.module.services.rest.InstructionServiceIT#testLifecycle(InstructionServiceIT.java:95)

我的 servlet 容器是 tomcat7.0.73 以下是我的 pom.xml 的片段,其中显示了我正在使用的其他版本的 software。

<dependency>
    <groupId>javax.servlet</groupId>
    <artifactId>javax.servlet-api</artifactId>
    <version>3.0.1</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>com.sun.jersey</groupId>
    <artifactId>jersey-json</artifactId>
    <version>1.9.1</version>
</dependency>
<dependency>
    <groupId>com.sun.jersey</groupId>
    <artifactId>jersey-server</artifactId>
    <version>1.9.1</version>
</dependency>
<dependency>
    <groupId>com.fasterxml.jackson.jaxrs</groupId>
    <artifactId>jackson-jaxrs-json-provider</artifactId>
    <version>2.8.3</version>
</dependency>
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.8.3</version>
</dependency>

感谢 Chris Hinshaw 为我指明了正确的方向。我使用的球衣版本不支持我使用的 Jackson 版本。我无法升级到新版本的 Jersey,所以我不得不从 Jackson 2 降级到 Jackson 1。 为了 reader.

的利益,我将旧内容注释掉了
  <!--<dependency>
    <groupId>com.fasterxml.jackson.jaxrs</groupId>
    <artifactId>jackson-jaxrs-json-provider</artifactId>
    <version>2.8.3</version>
  </dependency>
  <dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.8.4</version>
  </dependency>-->
  <dependency>
    <groupId>org.codehaus.jackson</groupId>
    <artifactId>jackson-all</artifactId>
    <version>1.9.0</version>
  </dependency>

显然,Jersey 1.9.1 与 Jackson 1.9.0 兼容,至少在我看来是这样。

我在 JsonMandateRequest 中的新导入看起来像这样。

import org.codehaus.jackson.annotate.JsonIgnoreProperties;
import org.codehaus.jackson.annotate.JsonProperty;
import org.codehaus.jackson.map.annotate.JsonDeserialize;
import org.codehaus.jackson.map.annotate.JsonSerialize;

希望这对某人有所帮助!