У меня есть REST API на основе Spring Boot + Hibernate + mySQL с обычными методами для операций CRUD и извлечения данных.В модели есть несколько много-много отношений, я покажу соответствующие классы для моей задачи:
Pectest.java
@Entity
@JsonIgnoreProperties({"hibernateLazyInitializer", "handler"})
@JsonIdentityInfo(generator=ObjectIdGenerators.PropertyGenerator.class, property="idtest")
public class Pectest {
@Id
@GeneratedValue
@Column(name="idtest")
private int idtest=-1;
private String testname;
private String testdescription;
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="idtesttype")
private Testtype testtype;
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="iddatatype")
private Datatype datatype;
private String testurl;
private Date datecreated;
private Date lastupdated;
@ManyToMany(cascade = {
CascadeType.PERSIST,
CascadeType.MERGE
})
@JoinTable(name = "test_checks_requisite",
joinColumns = @JoinColumn(name = "test_idtest"),
inverseJoinColumns = @JoinColumn(name = "requisite_idrequisite")
)
private Set<Requisite> requisites = new HashSet<Requisite>();
// Getters and setters
[...]
}
Requisite.java
@Entity
@JsonIgnoreProperties({"hibernateLazyInitializer", "handler"})
@JsonIdentityInfo(generator=ObjectIdGenerators.PropertyGenerator.class, property="idrequisite")
public class Requisite {
@Id
@GeneratedValue //(strategy=GenerationType.IDENTITY)
@Column(name="idrequisite")
private int idrequisite;
private String Name;
private String Title;
private String Description;
@ManyToOne(cascade = {CascadeType.PERSIST}, fetch=FetchType.LAZY)
@JoinColumn(name="functionality_idfunctionality")
private Functionality functionality;
@ManyToMany(mappedBy="requisites")
private Set<Pectest> tests = new HashSet<Pectest>();
// Getters and setters
[...]
}
Методы GET для обоих работают по назначению, корректно получая список Requisite данного Pectest .Однако, когда я пытаюсь обновить существующий Pectest , вызывая соответствующую точку входа API, я обнаружил, что она не получает существующие реквизиты в JSON из БД, давая им значение idrequisite 0 (как показано на рисунке).ниже).Поэтому Hibernate «решает», что они должны быть вставлены как новые.Сбой из-за ограничения уникальности в БД, но проблема в том, что они вообще не должны вставляться.
Процесс обновления Pectest объекта может включать (помимо очевидногоимя, описание и т. д., изменения) добавление дополнительных обязательных элементов в свой список, но эти элементы никогда не будут обновляться сами по себе.
Здесь содержится вся дополнительная информация:
Пример данных JSON, переданных в метод PUT API
{
"idtest": 9,
"testtype": { "idTestType": 5, "testTypeName": "Manual" },
"datatype": null,
"requisites": [
{
"idrequisite": 2,
"functionality": {
"idfunctionality": 1002,
"functionalityName": "Data Access OPC-UA",
"parentFunctionality": {
"idfunctionality": 1000,
"functionalityName": "OPC-UA",
"parentFunctionality": null,
"childrenFunctionalities": [
{
"idfunctionality": 1006,
"functionalityName": "Alarmas OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [],
"description": "Para el tratamiento de las Alarmas del producto BR, se utilizará el tipo propietario AlarmEventType, heredado del tipo básico de alarmas OffNormalAlarmType definido por la especificación OPC UA (Part 9). El Servidor OPC-UA deberá generar un nodo AlarmEventType para cada una de las alarmas definidas en el proyecto en concreto del producto BR (hasta 8192 alarmas distintas)."
},
{
"idfunctionality": 1001,
"functionalityName": "Atributos OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [],
"description": "Atributos"
},
1002,
{
"idfunctionality": 1009,
"functionalityName": "Device Integration OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [
{
"idfunctionality": 1011,
"functionalityName": "Tipo FieldDeviceType OPC-UA",
"parentFunctionality": 1009,
"childrenFunctionalities": [],
"description": "A partir del tipo estándar DeviceType definido por “OPC UA Device Integration” se crea el tipo propietario FieldDeviceType para representar e identificar al producto BR."
}
],
"description": "A partir del tipo estándar DeviceType definido por “OPC UA Device Integration” se crea el tipo propietario FieldDeviceType para representar e identificar al producto BR."
},
{
"idfunctionality": 1012,
"functionalityName": "Global Discovery Server OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [],
"description": "El servidor OPC UA implementará el “Global Certificate Management Server Facet” para comunicarse con el GDS "
},
{
"idfunctionality": 1008,
"functionalityName": "Historical Access OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [],
"description": "Esta funcionalidad es resuelta íntegramente por el Servidor OPC-UA, ya que el FW embebido no ofrece soporte para ella."
},
{
"idfunctionality": 1007,
"functionalityName": "Monitoring & Subscription OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [],
"description": "Subscripciones para visualización de datos desde un cliente OPC-UA"
},
{
"idfunctionality": 1010,
"functionalityName": "Seguridad OPC-UA",
"parentFunctionality": 1000,
"childrenFunctionalities": [],
"description": "El servidor OPC-UA debe implementar una serie de medidas de seguridad para garantizar la confidencialidad de la información."
}
],
"description": "OPC-UA Server"
},
"childrenFunctionalities": [
{
"idfunctionality": 1004,
"functionalityName": "AnalogType OPC-UA",
"parentFunctionality": 1002,
"childrenFunctionalities": [],
"description": "Este tipo propietario se utiliza para los nodos OPC-UA de variables no booleanas del producto BR que no sean parámetros."
},
{
"idfunctionality": 1003,
"functionalityName": "DigitalType OPC-UA",
"parentFunctionality": 1002,
"childrenFunctionalities": [],
"description": "Tipo propietario de datos booleanos servido por OPC-UA, que no sean parámetros."
},
{
"idfunctionality": 1005,
"functionalityName": "ParameterType OPC-UA",
"parentFunctionality": 1002,
"childrenFunctionalities": [],
"description": "Este tipo propietario se utiliza para los nodos OPC-UA de variables del producto BR que estén identificadas como parámetros. El servidor OPC-UA debe identificar dichas variables del producto BR como aquellas que tengan atributo ParamGID. "
}
],
"description": "Se ofrecerá acceso a las variables exportadas por el FW de la CPU del producto BR a través de tres tipos propietarios de nodos: DigitalType, AnalogType y ParameterType"
},
"name": "REQ-0001",
"description": "Se ofrece acceso a variables digitales a través de este tipo de nodo",
"title": "Digital Type"
},
{
"idrequisite": 4,
"functionality": 1002,
"name": "REQ-0003",
"description": "Se ofrece acceso a parámetros a través de este tipo de nodo",
"title": "Parameter Type"
},
{
"idrequisite": 3,
"functionality": 1002,
"name": "REQ-0002",
"description": "Se ofrece acceso a variables analógicas a través de este tipo de nodo",
"title": "Analog Type"
}
],
"testDescription": "El servidor es capaz de devolver correctamente todos los tipos de dato.",
"testURL": "",
"testName": "Lectura de DataTypes",
"dateCreated": "09-05-2018 08:57:27",
"lastUpdated": "09-05-2018 08:57:27"
}
PecTestController.java
[...]
@PostMapping(path="/test", produces= MediaType.APPLICATION_JSON_VALUE)
public Pectest createOrUpdateTest(@Valid @RequestBody Pectest newtest) {
try {
System.out.println("Create or update Test: " + objectmapper.writeValueAsString(newtest));
} catch (JsonProcessingException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return testService.saveOrUpdate(newtest);
}
Консольный вывод (в формате)
{
"idtest": 9,
"testtype": { "idTestType": 5, "testTypeName": "Manual" },
"datatype": null,
"requisites": [
{
"idrequisite": 0,
"functionality": {
"idfunctionality": 0,
"functionalityName": "Data Access OPC-UA",
"parentFunctionality": {
"idfunctionality": 0,
"functionalityName": "OPC-UA",
"parentFunctionality": null,
"childrenFunctionalities": [
{
"idfunctionality": 0,
"functionalityName": "Device Integration OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [
{
"idfunctionality": 0,
"functionalityName": "Tipo FieldDeviceType OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "A partir del tipo estándar DeviceType definido por “OPC UA Device Integration” se crea el tipo propietario FieldDeviceType para representar e identificar al producto BR."
}
],
"description": "A partir del tipo estándar DeviceType definido por “OPC UA Device Integration” se crea el tipo propietario FieldDeviceType para representar e identificar al producto BR."
},
{
"idfunctionality": 0,
"functionalityName": "Historical Access OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Esta funcionalidad es resuelta íntegramente por el Servidor OPC-UA, ya que el FW embebido no ofrece soporte para ella."
},
{
"idfunctionality": 0,
"functionalityName": "Atributos OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Atributos"
},
{
"idfunctionality": 0,
"functionalityName": "Alarmas OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Para el tratamiento de las Alarmas del producto BR, se utilizará el tipo propietario AlarmEventType, heredado del tipo básico de alarmas OffNormalAlarmType definido por la especificación OPC UA (Part 9). El Servidor OPC-UA deberá generar un nodo AlarmEventType para cada una de las alarmas definidas en el proyecto en concreto del producto BR (hasta 8192 alarmas distintas)."
},
0,
{
"idfunctionality": 0,
"functionalityName": "Monitoring & Subscription OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Subscripciones para visualización de datos desde un cliente OPC-UA"
},
{
"idfunctionality": 0,
"functionalityName": "Seguridad OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "El servidor OPC-UA debe implementar una serie de medidas de seguridad para garantizar la confidencialidad de la información."
},
{
"idfunctionality": 0,
"functionalityName": "Global Discovery Server OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "El servidor OPC UA implementará el “Global Certificate Management Server Facet” para comunicarse con el GDS "
}
],
"description": "OPC-UA Server"
},
"childrenFunctionalities": [
{
"idfunctionality": 0,
"functionalityName": "ParameterType OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Este tipo propietario se utiliza para los nodos OPC-UA de variables del producto BR que estén identificadas como parámetros. El servidor OPC-UA debe identificar dichas variables del producto BR como aquellas que tengan atributo ParamGID. "
},
{
"idfunctionality": 0,
"functionalityName": "DigitalType OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Tipo propietario de datos booleanos servido por OPC-UA, que no sean parámetros."
},
{
"idfunctionality": 0,
"functionalityName": "AnalogType OPC-UA",
"parentFunctionality": 0,
"childrenFunctionalities": [],
"description": "Este tipo propietario se utiliza para los nodos OPC-UA de variables no booleanas del producto BR que no sean parámetros."
}
],
"description": "Se ofrecerá acceso a las variables exportadas por el FW de la CPU del producto BR a través de tres tipos propietarios de nodos: DigitalType, AnalogType y ParameterType"
},
"name": "REQ-0003",
"description": "Se ofrece acceso a parámetros a través de este tipo de nodo",
"title": "Parameter Type"
},
{
"idrequisite": 0,
"functionality": 0,
"name": "REQ-0002",
"description": "Se ofrece acceso a variables analógicas a través de este tipo de nodo",
"title": "Analog Type"
},
{
"idrequisite": 0,
"functionality": 0,
"name": "REQ-0001",
"description": "Se ofrece acceso a variables digitales a través de este tipo de nodo",
"title": "Digital Type"
}
],
"testDescription": "El servidor es capaz de devolver correctamente todos los tipos de dato.",
"testURL": "",
"testName": "Lectura de DataTypes",
"dateCreated": null,
"lastUpdated": null
}
Как видите, идентификаторы для всех вложенных объектов равны нулю.Почему Hibernate не пытается получить эти элементы?В модели отсутствует какая-либо аннотация?
В качестве обходного пути я думаю об игнорировании всего списка Requisite объектов при обновлении Pectest и предоставлении отдельной точки входа в API для добавления реквизитовк тесту, который будет вызываться отдельно.
Я проверил этот поток , который указывает на этот пример, но не смог найти решение.