Hibernate - Persist @OneToOne с составным ключом - PullRequest
0 голосов
/ 04 июля 2018

У меня есть следующая таблица аудита, которая может содержать информацию аудита для любого другого класса моей схемы:

CREATE TABLE `audit` (
  `table_name` varchar(45) NOT NULL,
  `item_id` int(11) NOT NULL,
  `version` int(11) NOT NULL,
  `updated_at` datetime NOT NULL,
  `updated_by` varchar(25) NOT NULL,
  `comment` varchar(255) DEFAULT NULL,
  PRIMARY KEY (`table_name`,`item_id`,`version`)
)

Тогда в моей схеме есть разные сущности JPA, например:

@Entity(name = "EntityA")
@Table(name = "entity_a")
public class EntityA {
    @Id
    @GeneratedValue
    private Long id;

    private Long version;

    // Other fields

    @OneToOne(mappedBy = "id.item", targetEntity = EntityAAudit.class, fetch = FetchType.EAGER, cascade = CascadeType.ALL)
    private EntityAAudit audit;
}

В то же время у меня есть класс asbtract Audit, который является суперклассом для нескольких классов аудита для отдельных сущностей:

@MappedSuperclass
@Table(name = "audit")
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "table_name", discriminatorType = DiscriminatorType.STRING)
@DiscriminatorOptions(insert = true, force = true)
public abstract class AuditHistory {

    // Some audit fields like the date and the author of the modification

}

@Entity(name = "EntityAAudit")
@Table(name = "audit")
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorValue("entity_a")
@DiscriminatorOptions(insert = true, force = true)
public class EntityAAudit extends Audit {

    @EmbeddedId
    @JsonUnwrapped
    private AuditedId id;

    @Embeddable
    public static class AuditedId implements Serializable {
        @OneToOne
        @JoinColumn(name = "item_id", nullable = false)
        private EntityA item;

        @Column(name = "version", nullable = false)
        private Long version;
    }

}

Это сопоставление работает при извлечении сущностей и их информации аудита из базы данных, но не при вставке новой сущности с соответствующей информацией аудита:

EntityA entity = new EntityA();
entity.setVersion(/* ... */);
// Setting the other basic fields of the entity

EntityAAudit audit = new EntityAAudit();
// Setting the basic fields of the audit

entity.setAudit(audit);

entity = em.merge(entity);

Я получаю следующее исключение:

org.hibernate.id.IdentifierGenerationException: null id generated for:class EntityAAudit

Я буквально перепробовал все, что мог придумать или найти в Интернете, и в итоге все сводится к одной и той же проблеме: Hibernate пытается вставить мой объект Audit с пустыми значениями для item_id и version.

Если я вручную установлю свой экземпляр entity и версию в качестве идентификатора объекта audit следующим образом:

EntityA entity = new EntityA();
entity.setVersion(/* ... */);
// Setting the other basic fields of the entity

EntityAAudit audit = new EntityAAudit();
// Setting the basic fields of the audit
audit.setId(new EntityAAudit.AuditedId());
audit.getId().setItem(entity);
audit.getId().setVersion(entity.getVersion());

entity.setAudit(audit);

entity = em.merge(entity);

Тогда я получаю еще более неясную ошибку:

Caused by: java.lang.NullPointerException
    at org.hibernate.type.descriptor.java.AbstractTypeDescriptor.extractHashCode(AbstractTypeDescriptor.java:65)
    at org.hibernate.type.AbstractStandardBasicType.getHashCode(AbstractStandardBasicType.java:185)
    at org.hibernate.type.AbstractStandardBasicType.getHashCode(AbstractStandardBasicType.java:189)
    at org.hibernate.type.EntityType.getHashCode(EntityType.java:348)

Обратите внимание, что я не могу изменить ни структуру моей базы данных, ни версию Hibernate (5.1.0, я знаю, что в более поздних версиях исправлены некоторые ошибки, которые могли бы решить мою проблему ...).

Большое спасибо:)

1 Ответ

0 голосов
/ 04 июля 2018

Вы можете попробовать отображение «производной идентичности»:

@Entity(name = "EntityAAudit")
@Table(name = "audit")
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorValue("entity_a")
@DiscriminatorOptions(insert = true, force = true)
public class EntityAAudit extends Audit {

    @EmbeddedId
    @JsonUnwrapped
    private AuditedId id;

    @OneToOne
    @JoinColumn(name = "item_id", nullable = false)
    @MapsId("entityAId") // maps entityAId attribute of embedded id
    private EntityA item;

    @Embeddable
    public static class AuditedId implements Serializable {
        private Long entityAId; // corresponds to PK type of EntityA

        @Column(name = "version", nullable = false)
        private Long version;
    }

}

Обратите внимание на аннотацию @MapsId на EntityAAudit.item.

Кроме того, вам необходимо явно указать EntityAAudit.item и AuditedId.version. JPA волшебным образом не определяет и не устанавливает для вас какие-либо циклические ссылки.

Производные идентификаторы обсуждаются (с примерами) в спецификации JPA 2.2 в разделе 2.4.1.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...