Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

JsonException No _valueDeserializer assigned

I've a classic spring web app with some spring crud repository.

I'm trying to save my entities within a classic angular form and i'm getting randomly this error :

.w.s.m.s.DefaultHandlerExceptionResolver : Failed to read HTTP message: org.springframework.http.converter.HttpMessageNotReadableException: Could not read document: No _valueDeserializer assigned
at [Source: java.io.PushbackInputStream@11936b0f; line: 1, column: 19] (through reference chain: com.adeo.volt.domain.HalfFlow["exchange"]->com.adeo.volt.domain.Exchange["id"]); nested exception is com.fasterxml.jackson.databind.JsonMappingException: No _valueDeserializer assigned
at [Source: java.io.PushbackInputStream@11936b0f; line: 1, column: 19] (through reference chain: com.adeo.volt.domain.HalfFlow["exchange"]->com.adeo.volt.domain.Exchange["id"])

Here is my sended Json :

{
    "exchange": {
        "id": 305,
        "processExchange": [],
        "number": 305,
        "halfFlows": [],
        "name": "TEST2",
        "type": {
            "id": 2,
            "name": "real-time"
        },
        "technology": {
            "id": 1,
            "name": "ESB webMethods"
        },
        "exchangesObjectsExchanges": [{
            "id": 235,
            "exchangeObject": {
                "id": 180,
                "name": "TESTALEXX",
                "objectMeaning": "t",
                "internalReferent": "t",
                "internalBusinessReferent": null,
                "status": {
                    "id": 1,
                    "name": "Study In Progress"
                },
                "type": {
                    "id": 1,
                    "name": "Canonical"
                },
                "externalDocLink": null,
                "keywords": null,
                "qualityScore": null
            }
        }]
    },
    "way": {
        "id": 1,
        "name": "TO"
    },
    "halfFlowApplication": [{
        "application": {
            "id": 2,
            "name": "ADEXIOCRE2",
            "type": {
                "id": 1,
                "name": "INTERNAL"
            },
            "domain": {
                "id": 6,
                "name": "Logistics",
                "letter": "L"
            },
            "vendor": null,
            "projectName": null,
            "internalReferent": null,
            "groupApplicationType": {
                "id": 2,
                "name": "MULTI INSTANCE"
            },
            "halfFlowApplication": [{
                "id": 159
            }]
        }
    }],
    "halfFlowNumber": "L0305A",
    "name": "TEST2_TO_ADEXIOCRE2",
    "halfFlowBus": [{
        "bu": {
            "id": 10,
            "number": 10,
            "shortLabel": "BCIT",
            "longLabel": "BC ITALIE"
        },
        "id": null
    }],
    "connector": {
        "id": 4,
        "name": "FTP"
    }
}

Here is my classes :

Exchange.java

@Entity
public class Exchange {

@Id
@Column(unique = true, nullable = false)
@GenericGenerator(name = "ExchangeGenerator", strategy = "increment")
@GeneratedValue(generator = "ExchangeGenerator")
private Long id;

@Column(unique = true)
private String name;
private int number;
private long technologyId;
private long exchangeTypeId;

@OneToMany(mappedBy = "exchange")
private List<Halfflow> halfFlows;

public Exchange() {
}

/* Getters and Setters */
public Long getId() {
    return id;
}

public void setId(Long id) {
    this.id = id;
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}

public int getNumber() {
    return number;
}

public void setNumber(int number) {
    this.number = number;
}

public long getTechnologyId() {
    return technologyId;
}

public void setTechnologyId(long technologyId) {
    this.technologyId = technologyId;
}

public long getExchangeTypeId() {
    return exchangeTypeId;
}

public void setExchangeTypeId(Long exchangeTypeId) {
    this.exchangeTypeId = exchangeTypeId;
}

public void setExchangeTypeId(long exchangeTypeId) {
    this.exchangeTypeId = exchangeTypeId;
}

public List<Halfflow> getHalfFlows() {
    return halfFlows;
}

public void setHalfFlows(List<Halfflow> halfFlows) {
    this.halfFlows = halfFlows;
}

@Override
public String toString() {
    return "Exchange{" +
            "id=" + id +
            ", name='" + name + '\'' +
            ", number=" + number +
            ", technologyId=" + technologyId +
            '}';
}

}

Halfflow.java

@Entity
@Table(name="HALF_FLOW")
public class HalfFlow {

@Id
@Column(name = "ID", unique = true, nullable = false)
@GenericGenerator(name = "HalfflowGenerator", strategy = "increment")
@GeneratedValue(generator = "HalfflowGenerator")
@JsonProperty(value="id")
private Long id;

@Column(unique = true)
@JsonProperty(value="halfFlowNumber")
private String halfFlowNumber;

@JsonProperty(value="name")
private String name;

@ManyToOne
@JoinColumn(name = "WAY_ID")
private Way way;

@JsonProperty(value="externalDocLink")
private String externalDocLink;

@ManyToOne
@JoinColumn(name = "HALF_FLOW_MODE_ID")
@JsonProperty(value="mode")
private HalfFlowMode mode;

@JsonProperty(value="internalReferent")
private String internalReferent;

@JsonProperty(value="backupReferent")
private String backupReferent;

@ManyToOne
@JoinColumn(name = "DATA_FORMAT_ID")
@JsonProperty(value="format")
private DataFormat format;

@ManyToOne
@JoinColumn(name = "CONNECTOR_ID")
@JsonProperty(value="connector")
private Connector connector;

@ManyToOne
@JoinColumn(name = "CONFIDENTIALITY_ID")
@JsonProperty(value="confidentiality")
private Confidentiality confidentiality;

@ManyToOne
@JoinColumn(name = "INTEGRITY_ID")
@JsonProperty(value="integrity")
private Integrity integrity;

@ManyToOne
@JoinColumn(name="HALF_FLOW_AVAILABILITY_ID")
@JsonProperty(value="halfFlowAvailability")
private HalfFlowAvailability halfFlowAvailability;

@JsonIgnoreProperties(value = {"halfFlow"})
@OneToMany(mappedBy = "halfFlow")
@JsonProperty(value="halfFlowApplication")
private List<HalfFlowApplication> halfFlowApplication;

@OneToMany(mappedBy = "halfFlow")
@JsonProperty(value="halfFlowBus")
private List<HalfFlowBu> halfFlowBus;

@OneToMany(mappedBy = "halfFlow")
@JsonProperty(value="serviceHalfFlow")
private Set<ServiceHalfFlow> serviceHalfFlow;

@JsonIgnoreProperties(value = {"halfFlows"})
@ManyToOne
@JoinColumn(name="EXCHANGE_ID")
@JsonProperty(value="exchange")
private Exchange exchange;

@JsonProperty(value="bridge")
private boolean bridge;

public HalfFlow() {
}

public HalfFlow(String halfFlowNumber, String name) {
    this.halfFlowNumber = halfFlowNumber;
    this.name = name;
}

public HalfFlow(Long id, String halfFlowNumber, String name) {
    this.id = id;
    this.halfFlowNumber = halfFlowNumber;
    this.name = name;
}

@Override
public boolean equals(Object o) {
    if (this == o) return true;
    if (o == null || getClass() != o.getClass()) return false;

    HalfFlow halfFlow = (HalfFlow) o;

    if (id != null ? !id.equals(halfFlow.id) : halfFlow.id != null) return false;
    return name != null ? name.equals(halfFlow.name) : halfFlow.name == null;

}

@Override
public int hashCode() {
    int result = id != null ? id.hashCode() : 0;
    result = 31 * result + (name != null ? name.hashCode() : 0);
    return result;
}

public Long getId() {
    return id;
}

public void setId(Long id) {
    this.id = id;
}

public String getHalfFlowNumber() {
    return halfFlowNumber;
}

public void setHalfFlowNumber(String halfFlowNumber) {
    this.halfFlowNumber = halfFlowNumber;
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}

public Way getWay() {
    return way;
}

public void setWay(Way way) {
    this.way = way;
}

public String getExternalDocLink() {
    return externalDocLink;
}

public void setExternalDocLink(String externalDocLink) {
    this.externalDocLink = externalDocLink;
}

public HalfFlowMode getMode() {
    return mode;
}

public void setMode(HalfFlowMode mode) {
    this.mode = mode;
}

public String getInternalReferent() {
    return internalReferent;
}

public void setInternalReferent(String internalReferent) {
    this.internalReferent = internalReferent;
}

public String getBackupReferent() {
    return backupReferent;
}

public void setBackupReferent(String backupReferent) {
    this.backupReferent = backupReferent;
}

public DataFormat getFormat() {
    return format;
}

public void setFormat(DataFormat format) {
    this.format = format;
}

public Connector getConnector() {
    return connector;
}

public void setConnector(Connector connector) {
    this.connector = connector;
}

public Confidentiality getConfidentiality() {
    return confidentiality;
}

public void setConfidentiality(Confidentiality confidentiality) {
    this.confidentiality = confidentiality;
}

public Integrity getIntegrity() {
    return integrity;
}

public void setIntegrity(Integrity integrity) {
    this.integrity = integrity;
}

public HalfFlowAvailability getHalfFlowAvailability() {
    return halfFlowAvailability;
}

public void setHalfFlowAvailability(HalfFlowAvailability halfFlowAvailability) {
    this.halfFlowAvailability = halfFlowAvailability;
}

public List<HalfFlowApplication> getHalfFlowApplication() {
    return halfFlowApplication;
}

public void setHalfFlowApplication(List<HalfFlowApplication> halfFlowApplication) {
    this.halfFlowApplication = halfFlowApplication;
}

public Exchange getExchange() {
    return exchange;
}

public void setExchange(Exchange exchange) {
    this.exchange = exchange;
}

public Set<ServiceHalfFlow> getServiceHalfFlow() {
    return serviceHalfFlow;
}

public void setServiceHalfFlow(Set<ServiceHalfFlow> serviceHalfFlow) {
    this.serviceHalfFlow = serviceHalfFlow;
}

public List<HalfFlowBu> getHalfFlowBus() {
    if (halfFlowBus != null)
        Collections.sort(halfFlowBus);
    return halfFlowBus;
}

public void setHalfFlowBus(List<HalfFlowBu> halfFlowBus) {
    this.halfFlowBus = halfFlowBus;
}

public boolean isBridge() {
    return bridge;
}

public void setBridge(boolean bridge) {
    this.bridge = bridge;
}

@Override
public String toString() {
    return "HalfFlow{" +
            "id=" + id +
            ", halfFlowNumber='" + halfFlowNumber + '\'' +
            ", name='" + name + '\'' +
            ", way=" + way +
            ", externalDocLink='" + externalDocLink + '\'' +
            ", mode=" + mode +
            ", internalReferent='" + internalReferent + '\'' +
            ", backupReferent='" + backupReferent + '\'' +
            ", format=" + format +
            ", connector=" + connector +
            ", confidentiality=" + confidentiality +
            ", integrity=" + integrity+
            ", halfFlowAvailability=" + halfFlowAvailability +
            ", halfFlowApplication=" + halfFlowApplication +
            ", halfFlowBus=" + halfFlowBus +
            ", serviceHalfFlow=" + serviceHalfFlow +
            (exchange != null ? ", exchange=" + exchange.getName() : "exchange=null" )+
            ", bridge=" + bridge +
            '}';
}
}

Here is my controller signature :

@RequestMapping(method = RequestMethod.POST)
public ResponseEntity<HalfFlow> save(@RequestBody HalfFlow halfflow){...}

I'm wondering what can throw this error and this appear kind of randomly..

like image 555
RVA Avatar asked Nov 28 '22 22:11

RVA


2 Answers

As Alex said, to prevent the infinite recursion, you added this annotation

@JsonIgnoreProperties(value = {"exchange"})
@OneToMany(mappedBy = "exchange")
private List<HalfFlow> halfFlows;

So I actually solved this issue by adding the following to the annotation

@JsonIgnoreProperties(value = {"exchange"}, allowSetters = true)
@OneToMany(mappedBy = "exchange")
private List<HalfFlow> halfFlows;

This solution is better explained here https://softwareengineering.stackexchange.com/questions/300115/best-way-to-deal-with-hibernate-1-many-relationship-over-rest-json-service

like image 88
Justin Avatar answered Dec 04 '22 12:12

Justin


I think this issue occurs only when you serialize for the first time an Exchange entity. Maybe Jackson save a "copy" of attributes for each Object ?

To prevent infinite loop, you added this annotation :

@JsonIgnoreProperties(value = {"exchange"})
@OneToMany(mappedBy = "exchange")
private List<HalfFlow> halfFlows;

So, my guess is Jackson "blacklist" property exchange of Halfflow bean.

Workaround : Use an ObjectMapper for each entity on your controller :

ObjectMapper mapper = new ObjectMapper();

@RequestMapping(method = RequestMethod.POST)
public ResponseEntity<HalfFlow> save(@RequestBody String json) throws JsonParseException, JsonMappingException, IOException {
    HalfFlow node = mapper.readValue(json, HalfFlow.class);
    HalfFlow halfflow = mapper.convertValue(node, HalfFlow.class);
like image 24
Alex Avatar answered Dec 04 '22 12:12

Alex