org.jsimpledb.vaadin.QueryJObjectContainer Maven / Gradle / Ivy
Show all versions of jsimpledb-vaadin Show documentation
/*
* Copyright (C) 2015 Archie L. Cobbs. All rights reserved.
*/
package org.jsimpledb.vaadin;
import com.google.common.base.Function;
import com.google.common.collect.Iterators;
import java.util.Iterator;
import org.jsimpledb.CopyState;
import org.jsimpledb.JObject;
import org.jsimpledb.JSimpleDB;
import org.jsimpledb.JTransaction;
import org.jsimpledb.SnapshotJTransaction;
import org.jsimpledb.ValidationMode;
/**
* Vaadin {@link com.vaadin.data.Container} backed by {@link JSimpleDB} Java model objects acquired by performing
* a transactional query and copying out the results into a {@link SnapshotJTransaction}.
*
*
* The subclass method {@linkplain #queryForObjects queryForObjects()} performs the query returning the backing objects
* within an already-opened transaction. The {@link com.vaadin.data.Item}s in the container are backed by in-memory copies
* of the returned {@link JObject}s; these live inside a {@link SnapshotJTransaction} and therefore may persist indefinitely
* after the query transaction closes.
*
*
* Instances may be (re)loaded at any time by invoking {@link #reload}. During reload, the container opens a new
* {@link JTransaction}, queries for objects using {@link #queryForObjects queryForObjects()}, and copies each returned
* object into the associated {@link SnapshotJTransaction} via {@link #copyWithRelated copyWithRelated()},
* which copies the object and any related objects necessary for resolving container properties. The set of related
* objects associated with an object is determined by {@link #getRelatedObjects getRelatedObjects()}; by default,
* this is all directly referenced objects. Subclasses may override to customize.
*
*
* It is up to {@link #queryForObjects queryForObjects()} to determine what and how many objects are returned, their
* sort order, etc. Any objects returned by {@link #queryForObjects queryForObjects()} that are not instances of the
* container's {@linkplain #setType configured type} are ignored.
*
*
* {@link org.dellroad.stuff.vaadin7.ProvidesProperty @ProvidesProperty} Limitations
*
*
* The use of {@link org.dellroad.stuff.vaadin7.ProvidesProperty @ProvidesProperty} methods has certain implications.
* First, if the method reads any of the object's field(s) via their Java getter methods (as would normally be expected),
* this will trigger a schema upgrade of the object if needed; however, this schema upgrade will occur in the
* container's in-memory {@link SnapshotJTransaction} rather than in a real database transaction, so the
* {@link #VERSION_PROPERTY} will return a different schema version from what's in the database. The automatic schema
* upgrade can be avoided if desired by reading the field using the appropriate {@link JTransaction} field access method
* (e.g., {@link JTransaction#readSimpleField JTransaction.readSimpleField()}) and being prepared to handle a
* {@link org.jsimpledb.core.UnknownFieldException} if/when the object has an older schema version that does not contain
* the requested field.
*
*
* Secondly, because the values of reference fields (including complex sub-fields) are displayed using reference labels,
* and these are typically derived from the referenced object's fields, those indirectly referenced objects need to be
* copied into the container's {@link SnapshotJTransaction} as well. The easiest way to ensure these indirectly
* referenced objects are copied is by overriding {@link #getRelatedObjects getRelatedObjects()} as described above.
*/
@SuppressWarnings("serial")
public abstract class QueryJObjectContainer extends ReloadableJObjectContainer {
/**
* Constructor.
*
* @param jdb {@link JSimpleDB} database
* @param type type restriction, or null for no restriction
* @throws IllegalArgumentException if {@code jdb} is null
*/
protected QueryJObjectContainer(JSimpleDB jdb, Class> type) {
super(jdb, type);
}
/**
* (Re)load this container.
*
*
* This creates a new {@link JTransaction}, invokes {@link #queryForObjects} to query for backing objects,
* copies them into an in-memory {@link SnapshotJTransaction} via {@link #copyWithRelated copyWithRelated()},
* and builds the container from the result.
*/
@Override
public void reload() {
this.doInTransaction(new Runnable() {
@Override
public void run() {
QueryJObjectContainer.this.reloadInTransaction();
}
});
}
// This method runs within a transaction
private void reloadInTransaction() {
// Get objects from subclass
Iterator extends JObject> jobjs = this.queryForObjects();
// Copy objects (and their related object friends) into a snapshot transaction
final SnapshotJTransaction snapshotTx = JTransaction.getCurrent().createSnapshotTransaction(ValidationMode.DISABLED);
final CopyState copyState = new CopyState();
jobjs = Iterators.transform(jobjs, new Function() {
@Override
public JObject apply(JObject jobj) {
return QueryJObjectContainer.this.copyWithRelated(jobj, snapshotTx, copyState);
}
});
// Now actually load the objects
this.load(jobjs);
}
/**
* Copy the given database object, and any related objects needed by any
* {@link org.dellroad.stuff.vaadin7.ProvidesProperty @ProvidesProperty}-annotated methods,
* into the specified transaction.
*
*
* The implementation in {@link JObjectContainer} copies {@code jobj}, and all of {@code jobj}'s related objects returned
* by {@link #getRelatedObjects getRelatedObjects()}, via {@link JTransaction#copyTo(JTransaction, CopyState, Iterable)}.
*
* @param target the object to copy, or null (ignored)
* @param dest destination transaction
* @param copyState tracks what's already been copied
* @return the copy of {@code target} in {@code dest}, or null if {@code target} is null
* @see #getRelatedObjects getRelatedObjects()
*/
public JObject copyWithRelated(JObject target, JTransaction dest, CopyState copyState) {
// Ignore null
if (target == null)
return null;
// Copy out target object
final JTransaction jtx = target.getTransaction();
final JObject copy = target.copyTo(dest, null, copyState);
// Copy out target's related objects
final Iterable extends JObject> relatedObjects = this.getRelatedObjects(target);
if (relatedObjects != null)
jtx.copyTo(dest, copyState, relatedObjects);
// Done
return copy;
}
/**
* Find objects related to the specified object that are needed by any
* {@link org.dellroad.stuff.vaadin7.ProvidesProperty @ProvidesProperty}-annotated
* methods.
*
*
* This defines all of the other objects on which any container property of {@code jobj} may depend.
* These related objects will copied along with {@code obj} into the container's
* {@link SnapshotJTransaction} when the container is (re)loaded.
*
*
* The implementation in {@link JObjectContainer} returns all objects that are directly referenced by {@code jobj},
* delegating to {@link JSimpleDB#getReferencedObjects JSimpleDB.getReferencedObjects()}.
* Subclasses may override this method to refine the selection.
*
* @param jobj the object being copied
* @return {@link Iterable} of additional objects to be copied, or null for none; any null values are ignored
* @throws IllegalArgumentException if {@code jobj} is null
*/
protected Iterable extends JObject> getRelatedObjects(JObject jobj) {
return this.jdb.getReferencedObjects(jobj);
}
/**
* Query for the database objects that will be used to fill this container. Objects should be returned in the
* desired order; duplicates and null values will be ignored.
*
*
* A {@link JTransaction} will be open in the current thread when this method is invoked.
*
* @return database objects
*/
protected abstract Iterator extends JObject> queryForObjects();
}