Ошибка после удаления при обновлении в одной транзакции (удаленный объект будет повторно сохранен каскадом)

Я использую hibernate 4 в качестве провайдера jpa. В своем коде я вставляю новую версию отчета (в PactRegister), выбираю все версии этого отчета, удаляю дочерние записи (ActRegister) и вычисляю текущую активную версию отчета с помощью вставки дочерней записи. Но действие удаляет дочерний элемент и вставляет дочерний элемент для того же исключения родительской записи. Что не так?

И у меня есть сообщение об ошибке (удаленный объект будет повторно сохранен каскадом):

 ru.sigmasoft.DoMessage.Factory.State.OracleProcess.OracleSenderHandler.handle(OracleSenderHandler.java:73)
            at ru.sigmasoft.DoMessage.Factory.State.OracleProcess.OracleProcessingTemplate.process(OracleProcessingTemplate.java:63)
            at ru.sigmasoft.DoMessage.Oracle.OracleTableListenerImpl.onMessage(OracleTableListenerImpl.java:28)
            at ru.sigmasoft.DoMessage.Oracle.OracleTableConsumer.listen(OracleTableConsumer.java:68)
            at ru.sigmasoft.DoMessage.Oracle.OracleTableConsumer.setMessageListener(OracleTableConsumer.java:35)
            at ru.DbManage.run(DbManage.java:22)
            at java.lang.Thread.run(Thread.java:619)
        Caused by: java.lang.IllegalArgumentException: org.hibernate.ObjectDeletedException: deleted object would be re-saved by cascade (remove deleted object from associations): [ru.sigmasoft.DOSrv.Server.domain.VersionSchema.ActRegister#49]
            at org.hibernate.ejb.AbstractEntityManagerImpl.merge(AbstractEntityManagerImpl.java:882)
            at ru.sigmasoft.DOSrv.Server.service.jpa.oraDb.Repository_PactRegister.modify(Repository_PactRegister.java:53)
            at ru.sigmasoft.DOSrv.Server.service.bean.Registration.Registration_Act.registerDocumentVersion(Registration_Act.java:62)
            at ru.sigmasoft.Specification.Message.CMN13011_Handler.handle(CMN13011_Handler.java:74)
            at ru.sigmasoft.DoMessage.Factory.State.OracleProcess.ParseBody.handle(ParseBody.java:107)
            at ru.sigmasoft.DoMessage.Factory.State.OracleProcess.OracleProcessingTemplate.process(OracleProcessingTemplate.java:46)
            ... 5 more
        Caused by: org.hibernate.ObjectDeletedException: deleted object would be re-saved by cascade (remove deleted object from associations): [ru.sigmasoft.DOSrv.Server.domain.VersionSchema.ActRegister#49]
            at org.hibernate.internal.SessionImpl.forceFlush(SessionImpl.java:1222)
            at org.hibernate.event.internal.AbstractSaveEventListener.performSave(AbstractSaveEventListener.java:177)
            at org.hibernate.event.internal.AbstractSaveEventListener.saveWithGeneratedId(AbstractSaveEventListener.java:136)
            at org.hibernate.ejb.event.EJB3MergeEventListener.saveWithGeneratedId(EJB3MergeEventListener.java:71)
            at org.hibernate.event.internal.DefaultMergeEventListener.saveTransientEntity(DefaultMergeEventListener.java:236)
            at org.hibernate.event.internal.DefaultMergeEventListener.entityIsTransient(DefaultMergeEventListener.java:216)
            at org.hibernate.event.internal.DefaultMergeEventListener.onMerge(DefaultMergeEventListener.java:154)
            at org.hibernate.internal.SessionImpl.fireMerge(SessionImpl.java:910)
            at org.hibernate.internal.SessionImpl.merge(SessionImpl.java:892)
            at org.hibernate.engine.spi.CascadingAction$6.cascade(CascadingAction.java:288)
            at org.hibernate.engine.internal.Cascade.cascadeToOne(Cascade.java:380)
            at org.hibernate.engine.internal.Cascade.cascadeAssociation(Cascade.java:323)
            at org.hibernate.engine.internal.Cascade.cascadeProperty(Cascade.java:208)
            at org.hibernate.engine.internal.Cascade.cascade(Cascade.java:165)
            at org.hibernate.event.internal.DefaultMergeEventListener.cascadeOnMerge(DefaultMergeEventListener.java:439)
            at org.hibernate.event.internal.DefaultMergeEventListener.entityIsPersistent(DefaultMergeEventListener.java:183)
            at org.hibernate.event.internal.DefaultMergeEventListener.onMerge(DefaultMergeEventListener.java:157)
            at org.hibernate.event.internal.DefaultMergeEventListener.onMerge(DefaultMergeEventListener.java:76)
            at org.hibernate.internal.SessionImpl.fireMerge(SessionImpl.java:900)
            at org.hibernate.internal.SessionImpl.merge(SessionImpl.java:884)
            at org.hibernate.internal.SessionImpl.merge(SessionImpl.java:888)
            at org.hibernate.ejb.AbstractEntityManagerImpl.merge(AbstractEntityManagerImpl.java:879)

Моя сущность:

       @Entity
    @Table(name = "PACT_REGISTER")
    @XmlRootElement
    public class PactRegister implements Serializable {

        private static final long serialVersionUID = 1L;
        // @Max(value=?)  @Min(value=?)//if you know range of your decimal fields consider using these annotations to enforce field validation
        @Id
        @GenericGenerator(name = "generatorPactR", strategy = "foreign",
                parameters =
                @Parameter(name = "property", value = "edRegister"))
        @GeneratedValue(generator = "generatorPactR")
        @Basic(optional = false)
        @Column(name = "ID")
        private Long id;
        @Column(name = "G071")
        private String g071;
        @Column(name = "G072")
        @Temporal(TemporalType.DATE)
        private Date g072;
        @Column(name = "G073")
        private String g073;
        @Column(name = "G073K")
        private String g073k;
        @Column(name = "SVHLICK")
        private String svhlick;
        @Column(name = "SVHLIC")
        private String svhlic;
        @Column(name = "MODIFICATIONDATETIME")
        @Temporal(TemporalType.DATE)
        private Date modificationdatetime;
        @JoinColumn(name = "ID", referencedColumnName = "ID", insertable = false, updatable = false)
        @OneToOne(optional = false, fetch = FetchType.LAZY)
        private EdRegister edRegister;
        @OneToOne(cascade = CascadeType.ALL, orphanRemoval = true , mappedBy = "pactRegister", fetch = FetchType.LAZY)
        private ActRegister actRegister;

        public PactRegister() {
        }

        public PactRegister(Long id) {
            this.id = id;
        }

        public Long getId() {
            return id;
        }

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

        public String getG071() {
            return g071;
        }

        public void setG071(String g071) {
            this.g071 = g071;
        }

        public Date getG072() {
            return g072;
        }

        public void setG072(Date g072) {
            this.g072 = g072;
        }

        public String getG073() {
            return g073;
        }

        public void setG073(String g073) {
            this.g073 = g073;
        }

        public String getG073k() {
            return g073k;
        }

        public void setG073k(String g073k) {
            this.g073k = g073k;
        }

        public String getSvhlick() {
            return svhlick;
        }

        public void setSvhlick(String svhlick) {
            this.svhlick = svhlick;
        }

        public String getSvhlic() {
            return svhlic;
        }

        public void setSvhlic(String svhlic) {
            this.svhlic = svhlic;
        }

        public Date getModificationdatetime() {
            return modificationdatetime;
        }

        public void setModificationdatetime(Date modificationdatetime) {
            this.modificationdatetime = modificationdatetime;
        }

        public EdRegister getEdRegister() {
            return edRegister;
        }

        public void setEdRegister(EdRegister edRegister) {
            this.edRegister = edRegister;
        }

        public ActRegister getActRegister() {
            return actRegister;
        }

        protected void setActRegister(ActRegister actRegister) {
            this.actRegister = actRegister;
        }

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

        @Override
        public boolean equals(Object object) {
            // TODO: Warning - this method won't work in the case the id fields are not set
            if (!(object instanceof PactRegister)) {
                return false;
            }
            PactRegister other = (PactRegister) object;
            if ((this.id == null && other.id != null) || (this.id != null && !this.id.equals(other.id))) {
                return false;
            }
            return true;
        }

        @Override
        public String toString() {
            return "Eh_OWNER.PactRegister[ id=" + id + " ]";
        }

        public void removeFromActive(final ActRegister ar) {
            // check for no-op
            if (ar == null /*|| !getActRegister().contains(ar)*/) {
                return;
            }
            // dissociate arg
            //  actRegister=null;
            this.setActRegister(null);
            ar.setPactRegister(null);

            // additional business logic
            //onRemoveFromActive(ar);

        }

        public void addToActive(final ActRegister ar) {
            // check for no-op
            if (ar == null /*|| !getActRegister().contains(ar)*/) {
                return;
            }
            // dissociate arg
            //  actRegister=null;
            setActRegister(ar);
            ar.setPactRegister(this);

            // additional business logic
            //onRemoveFromActive(ar);

        }
and

    @Entity
    @Table(name = "ACT_REGISTER")
    @XmlRootElement
    public class ActRegister implements Serializable {

        private static final long serialVersionUID = 1L;
        // @Max(value=?)  @Min(value=?)//if you know range of your decimal fields consider using these annotations to enforce field validation
        @Id
        @GenericGenerator(name = "generatorActR", strategy = "foreign",
                parameters =
                @Parameter(name = "property", value = "pactRegister"))
        @GeneratedValue(generator = "generatorActR")
        @Basic(optional = false)
        @Column(name = "ID")
        private Long id;
        @JoinColumn(name = "ID", referencedColumnName = "ID", insertable = false, updatable = false)
        @OneToOne(optional = false, fetch = FetchType.LAZY)
        private PactRegister pactRegister;

        public ActRegister() {
        }

        public ActRegister(Long id) {
            this.id = id;
        }

        public Long getId() {
            return id;
        }

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

        public PactRegister getPactRegister() {
            return pactRegister;
        }

        public void setPactRegister(PactRegister pactRegister) {
            this.pactRegister = pactRegister;
        }

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

        @Override
        public boolean equals(Object object) {
            // TODO: Warning - this method won't work in the case the id fields are not set
            if (!(object instanceof ActRegister)) {
                return false;
            }
            ActRegister other = (ActRegister) object;
            if ((this.id == null && other.id != null) || (this.id != null && !this.id.equals(other.id))) {
                return false;
            }
            return true;
        }

        @Override
        public String toString() {
            return "Eh_OWNER.ActRegister[ id=" + id + " ]";
        }
    }

мой репозиторий:

    public class Repository_PactRegister extends GenericRepositoryImpl<PactRegister, Long> {

        public Repository_PactRegister(EntityManager em) {
            super(em);
        }

        public List<PactRegister> findActive(String g071, Date g072, String g073, String g073k, String svhLick, String svhLic) {
            String SELECT_ACTIVE = "select * from PACT_REGISTER a inner join ACT_REGISTER b on a.id=b.id \n"
                    + " where a.g071=:G071 and a.g072=:G072 and a.g073=:G073 and a.g073k=:G073K and a.SVHLICK=:SVHLICK and a.SVHLIC=:SVHLIC";
            Query q = em.createNativeQuery(SELECT_ACTIVE, PactRegister.class);
            q.setParameter("G071", g071);
            q.setParameter("G072", g072, TemporalType.DATE);
            q.setParameter("G073", g073);
            q.setParameter("G073K", g073k);
            q.setParameter("SVHLICK", svhLick);
            q.setParameter("SVHLIC", svhLic);
            q.setParameter("SVHLIC", svhLic);
            List<PactRegister> results = q.getResultList();
            return results;
        }

        public PactRegister findLastByDocIdentifyer(String g071, Date g072, String g073, String g073k, String svhLick, String svhLic) {
            //Поиск записи с максимальной датой модификации для данного отчета
            String SELECT_MAX = "select * from PACT_REGISTER f where f.id=(select max(id) \n"
                    + " from PACT_REGISTER a)";
            Query q = em.createNativeQuery(SELECT_MAX, PactRegister.class);
            List<PactRegister> results = q.getResultList();
            return results.isEmpty() ? null : results.get(0);

        }

        @Override
        public PactRegister modify(PactRegister entity) {
            em.find(PactRegister.class, entity.getId());
            PactRegister mergedPactRegister = em.merge(entity);
            return mergedPactRegister;
        }
    }

Логика: Следующее действие выполняется в одной транзакции:

        private final Repository_EdRegister rer;
        private final Repository_PactRegister rpr;
         ......
          public PactRegister registerDocumentVersion(Act doc, EdRegister edRegister) {
            //insert new record of report
            PactRegister pactRegister = new PactRegister();
            pactRegister.setG071(doc.getG071());
            pactRegister.setG072(doc.getG072());
            pactRegister.setG073(doc.getG073());
            pactRegister.setG073k("0"/*doc.getG073k()*/);
            pactRegister.setSvhlick(doc.getSvhlick());
            pactRegister.setSvhlic(doc.getSvhlic());
            pactRegister.setModificationdatetime(new Date()/*doc.getModificationdatetime()*/);//Lkz ncnbhjdfybz!!
            edRegister.addToPactRegister(pactRegister);
            rpr.create(pactRegister);
    // Get parent records for this report
            List<PactRegister> activeS = rpr.findActive(doc.getG071(), doc.getG072(), doc.getG073(), /*doc.getG073k()*/ "0", doc.getSvhlick(), doc.getSvhlic());
    //remove all childs
            if (pactRegister != null) {
                for (PactRegister active : activeS) {

                    ActRegister ar = active.getActRegister();
                    active.removeFromActive(ar);
                    //   ar = null;
                    rpr.modify(active);

                }
            }


    //Choose active record
            // PactRegister activeCandidate = rpr.findById(PactRegister.class, Long.valueOf("110"));
            PactRegister activeCandidate = rpr.findLastByDocIdentifyer(doc.getG071(), doc.getG072(), doc.getG073(), "0"/*doc.getG073k()*/, doc.getSvhlick(), doc.getSvhlic());
            activeCandidate.addToActive(new ActRegister());
            rpr.modify(activeCandidate);
            return activeCandidate;

    }Exception throws when transaction commit. And only if execute method rpr.findLastByDocIdentifyer (it's call   

     String SELECT_MAX = "select * from PACT_REGISTER f where f.id=(select max(id) \n"
                    + " from PACT_REGISTER a)";
            Query q = em.createNativeQuery(SELECT_MAX, PactRegister.class);
            List<PactRegister> results = q.getResultList();
            return results.isEmpty() ? null : results.get(0);

   )

0 ответов

Другие вопросы по тегам