blob: 200d1e7274f5213b78da39b356a71a286b8f9827 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2011-2012 Oracle. All rights reserved.
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v1.0 and Eclipse Distribution License v. 1.0
* which accompanies this distribution.
* The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v10.html
* and the Eclipse Distribution License is available at
* http://www.eclipse.org/org/documents/edl-v10.php.
*
* Contributors:
* dclarke - Bug 361016: Future Versions Examples
******************************************************************************/
package temporal;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.Set;
import java.util.Vector;
import javax.persistence.EntityManager;
import org.eclipse.persistence.descriptors.ClassDescriptor;
import org.eclipse.persistence.descriptors.changetracking.ChangeTracker;
import org.eclipse.persistence.internal.sessions.AbstractSession;
import org.eclipse.persistence.internal.sessions.RepeatableWriteUnitOfWork;
import org.eclipse.persistence.mappings.DatabaseMapping;
import org.eclipse.persistence.mappings.ForeignReferenceMapping;
import temporal.persistence.DescriptorHelper;
/**
* Apply an {@link EditionSet} making all of its contained editions the current.
* This involves copying all relevant values into the current including the
* {@link Effectivity} and then delete this edition. The OID of the current
* should be changed to that of the edition when the operation is complete.
*
* @author dclarke
* @since EclipseLink 2.3.1
*/
public class EditionSetHelper {
/**
* Apply the {@link EditionSet} causing all its editions to become the
* continuity.
*
* @param em
* @param editionSet
*/
public static void apply(EntityManager em, EditionSet editionSet) {
for (EditionSetEntry ese : editionSet.getEntries()) {
copyValues(em, ese);
}
em.remove(editionSet);
}
public static void copyValues(EntityManager em, EditionSetEntry entry) {
TemporalEntity<?> edition = entry.getTemporalEntity();
TemporalEntity<?> continuity = entry.getTemporalEntity().getContinuity();
AbstractSession session = em.unwrap(RepeatableWriteUnitOfWork.class);
ClassDescriptor descriptor = DescriptorHelper.getCurrentDescriptor(session, edition.getClass());
for (String attr : entry.getAttributes()) {
DatabaseMapping mapping = descriptor.getMappingForAttributeName(attr);
if (!mapping.isForeignReferenceMapping()) {
Object value = mapping.getRealAttributeValueFromObject(edition, session);
Object oldValue = mapping.getRealAttributeValueFromObject(continuity, session);
mapping.setRealAttributeValueInObject(continuity, value);
if (continuity instanceof ChangeTracker) {
PropertyChangeListener listener = ((ChangeTracker) continuity)._persistence_getPropertyChangeListener();
listener.propertyChange(new PropertyChangeEvent(continuity, attr, oldValue, value));
}
}
}
// continuity.applyEdition(edition);
continuity.getEffectivity().setEnd(edition.getEffectivity().getEnd());
}
/**
* Move the provided {@link EditionSet} to the new effective time
*
* @throws IllegalArgumentException
* for invalid effective time or mismatched
* {@link TemporalEntityManager} and {@link EditionSet}
*/
public static void move(TemporalEntityManager em, long effective) {
if (effective <= Effectivity.BOT) {
throw new IllegalArgumentException("Invalid effective time for move: " + effective);
}
if (!em.hasEditionSet()) {
em.setEffectiveTime(effective);
return;
}
RepeatableWriteUnitOfWork uow = em.unwrap(RepeatableWriteUnitOfWork.class);
EditionSet es = em.getEditionSet();
// Need to ensure temporal objects are loaded before changing the
// effective time.
for (EditionSetEntry entry : es.getEntries()) {
entry.getTemporal();
}
EditionSet newES = new EditionSet(effective);
em.persist(newES);
em.setEditionSet(newES);
for (EditionSetEntry entry : es.getEntries()) {
// Look through relationship mappings for references to temporal
// which do not exist at the new effective time
ClassDescriptor descriptor = DescriptorHelper.getEditionDescriptor(uow, entry.getTemporal().getClass());
Set<ForeignReferenceMapping> mappings = DescriptorHelper.getTemporalMappings(descriptor);
for (ForeignReferenceMapping mapping : mappings) {
Temporal currentTarget = (Temporal) mapping.getRealAttributeValueFromObject(entry.getTemporal(), uow);
if (currentTarget != null) {
if (currentTarget.getEffectivity().getStart() > effective) {
// TODO: What about pending changes?
uow.getIdentityMapAccessor().invalidateObject(currentTarget);
}
Object id = ((TemporalEntity<?>) currentTarget).getContinuityId();
if (id instanceof Object[] || id instanceof Vector) {
throw new RuntimeException("Composite Key not supported");
}
@SuppressWarnings("unchecked")
Temporal target = em.find(mapping.getReferenceClass(), id);
if (target == null) {
throw new IllegalStateException();
}
mapping.setRealAttributeValueInObject(entry.getTemporal(), target);
} else {
// TODO
//throw new IllegalStateException();
}
}
entry.getTemporal().getEffectivity().setStart(effective);
newES.getEntries().add(entry);
entry.setEditionSet(newES);
}
es.getEntries().clear();
em.remove(es);
}
}