org.tentackle.persist.AbstractDbObject Maven / Gradle / Ivy
/**
* Tentackle - http://www.tentackle.org
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package org.tentackle.persist;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.io.Serializable;
import java.rmi.RemoteException;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.function.Supplier;
import org.tentackle.app.AbstractApplication;
import org.tentackle.log.Logger;
import org.tentackle.log.Logger.Level;
import org.tentackle.log.LoggerFactory;
import org.tentackle.misc.IdSerialTuple;
import org.tentackle.misc.Identifiable;
import org.tentackle.misc.Immutable;
import org.tentackle.misc.ImmutableException;
import org.tentackle.misc.SerialNumbered;
import org.tentackle.pdo.NotFoundException;
import org.tentackle.pdo.PdoTracker;
import org.tentackle.pdo.PersistenceException;
import org.tentackle.pdo.Persistent;
import org.tentackle.pdo.Session;
import org.tentackle.pdo.SessionDependable;
import org.tentackle.pdo.SessionHolder;
import org.tentackle.persist.rmi.AbstractDbObjectRemoteDelegate;
import org.tentackle.persist.rmi.DbObjectResult;
import org.tentackle.sql.Backend;
import static org.tentackle.sql.Backend.SQL_ALLSTAR;
import static org.tentackle.sql.Backend.SQL_AND;
import static org.tentackle.sql.Backend.SQL_COMMA;
import static org.tentackle.sql.Backend.SQL_DELETE;
import static org.tentackle.sql.Backend.SQL_EQUAL;
import static org.tentackle.sql.Backend.SQL_EQUAL_PAR;
import static org.tentackle.sql.Backend.SQL_FROM;
import static org.tentackle.sql.Backend.SQL_ORDERBY;
import static org.tentackle.sql.Backend.SQL_SELECT;
import static org.tentackle.sql.Backend.SQL_SET;
import static org.tentackle.sql.Backend.SQL_UPDATE;
import static org.tentackle.sql.Backend.SQL_WHERE;
import static org.tentackle.sql.Backend.SQL_WHEREALL;
/**
* A persistent low-level database object.
*
* All persistent objects must extend AbstractDbObject, which provides the generic
* functionality of persistent objects. Every AbstractDbObject is associated
* to a logical {@link Session}, which can be either local or remote.
* The application-specific configuration is achieved by implementing and/or
* overriding methods (pure OO-approach). These methods are generated
* by wurblets (see http://www.wurbelizer.org) which are already provided by Tentackle.
*
* AbstractDbObjects have the following predefined attributes:
*
* -
* id: a unique number (long) that identifies the object. The number is
* guaranteed to be unique for all objects of the same class.
*
* -
* serial: the object's serial number. When an object is first
* persisted it gets the serial 1. Every update increments the serial. The serial
* is primarily used for optimistic locking and cache synchronization.
*
* -
* tableSerial: a serial number for all objects of this class
* (per table or relation in database lingo). This attribute is optional and
* allows to determine which objects have been modified/inserted/deleted since
* the last change to any object within the same class.
*
* -
* classId: a class id, if this class is a leaf in a single- or multi-table inheritance tree.
*
*
*
* @param the persistent class type
*/
public abstract class AbstractDbObject
>
implements Immutable, Identifiable, SerialNumbered, SessionDependable, Serializable, Comparable
, Cloneable {
/**
* Instantiates a new db object for a given class without a session.
*
* @param
the class type
* @param clazz the class
* @return the object
*/
static public
> P newInstance(Class
clazz) {
try {
// load the class
return clazz.newInstance();
}
catch (InstantiationException | IllegalAccessException e) {
throw new PersistenceException("creating object for " + clazz + " failed", e);
}
}
/**
* Instantiates a new object for a given class and session.
*
* @param
the class type
* @param session the session
* @param clazz the class
* @return the object
*/
static public
> P newInstance(Session session, Class
clazz) {
// load the class
P obj = newInstance(clazz);
obj.setSession(session);
return obj;
}
/** name of ID column. */
public static final String CN_ID = org.tentackle.common.Constants.CN_ID;
/** name of ID attribute. */
public static final String AN_ID = org.tentackle.common.Constants.AN_ID;
/** name of serial column. */
public static final String CN_SERIAL = org.tentackle.common.Constants.CN_SERIAL;
/** name of serial attribute. */
public static final String AN_SERIAL = org.tentackle.common.Constants.AN_SERIAL;
/** name of tableserial column. */
public static final String CN_TABLESERIAL = org.tentackle.common.Constants.CN_TABLESERIAL;
/** name of tableserial attribute. */
public static final String AN_TABLESERIAL = org.tentackle.common.Constants.AN_TABLESERIAL;
/** name of class-ID column. */
public static final String CN_CLASSID = org.tentackle.common.Constants.CN_CLASSID;
/** name of class-ID attribute. */
public static final String AN_CLASSID = org.tentackle.common.Constants.AN_CLASSID;
// avoids excessive creation of comparators
/** ID comparator **/
public static IdComparator> idComparator = new IdComparator<>();
/** name comparator **/
public static NameComparator> nameComparator = new NameComparator<>();
/** name + ID comparator **/
public static NameIdComparator> nameIdComparator = new NameIdComparator<>();
/** transaction name: insert plain **/
public static final String TX_INSERT_PLAIN = "insert plain";
/** transaction name: insert object **/
public static final String TX_INSERT_OBJECT = "insert object";
/** transaction name: update plain **/
public static final String TX_UPDATE_PLAIN = "update plain";
/** transaction name: dummy update **/
public static final String TX_DUMMY_UPDATE = "dummy update";
/** transaction name: update serial **/
public static final String TX_UPDATE_SERIAL = "update serial";
/** transaction name: update serial and tableserial **/
public static final String TX_UPDATE_SERIAL_AND_TABLESERIAL = "update serial and tableserial";
/** transaction name: update tableserial **/
public static final String TX_UPDATE_TABLESERIAL = "update tableserial";
/** transaction name: update object **/
public static final String TX_UPDATE_OBJECT = "update object";
/** transaction name: saveObject **/
public static final String TX_SAVE = "save";
/** transaction name: sync **/
public static final String TX_SYNC = "sync";
/** transaction name: delete object **/
public static final String TX_DELETE_OBJECT = "delete object";
/** transaction name: saveObject list **/
public static final String TX_SAVE_LIST = "save list";
/** transaction name: delete list **/
public static final String TX_DELETE_LIST = "delete list";
/** transaction name: delete missing in list **/
public static final String TX_DELETE_MISSING_IN_LIST = "delete missing in list";
/**
* logger for this class.
*/
private static final Logger LOGGER = LoggerFactory.getLogger(AbstractDbObject.class);
private static final long serialVersionUID = 3452560056309085969L;
// data object attributes are always private
@Persistent("object id")
private long id; // unique Object ID
@Persistent("object version")
private long serial; // serial-nummer (version to detect simultaneous updates)
@Persistent("class id")
private int classId; // the class id (overrides classvariables classid)
@Persistent("table version")
private long tableSerial; // last table serial from countModification (only if isTableSerialProvided() == true)
// not persistable (but not transient due to RMI)
private boolean modified; // true if object is modified and not written to db yet
private boolean immutable; // true if object is immutable
private Level immutableLoggingLevel; // optional logging for immutable violations instead of exception
protected boolean preparePending; // true if prepareDelete() or prepareSave() needs to be invoked
// transient (not transferred to/from the application server)
private transient Db session; // database session (transient because it shouldn't be serialized)
private transient boolean sessionImmutable; // true if Db is immutable
private SessionHolder sessionHolder; // a redirector to retrieve the session
private transient boolean overloadable; // true if object can be loaded more than once from the database
// if this object is being replayed from a modification log
private transient ModificationLog modlog;
// for PropertyChangeListeners
private transient volatile PropertyChangeSupport changeSupport;
/**
* Creates a database object.
*
* @param db the logical {@link Db}-connection
*/
public AbstractDbObject(Db db) {
setSession(db);
}
/**
* Creates a database object not associated to a logical {@link Db}-connection.
* The connection must be set via {@link #setSession} in order to use it.
*/
public AbstractDbObject() {
// nothing to do
}
/**
* Gets the default string value.
* The default implementation invokes {@link #toGenericString}.
*
* @return the string value of this AbstractDbObject
*/
@Override
public String toString() {
return toGenericString();
}
/**
* Gets the string value: "<className>[id/serial]".
*
* Example: {@code "de.krake.plsbl.Product[344/2]"}
*
* @return the string value of this object
*/
@Override
public final String toGenericString() {
return getClass().getName() +
'[' +
id +
'/' +
serial +
']';
}
/**
* Gets the ID string: "classId:id".
* The ID string describes the PDO by its classId and object-ID.
*
* Example: {@code "1022:1258474"}
*
* @return the id string
*/
public final String toIdString() {
return getClassId() + ":" + getId();
}
/**
* Gets the some attributes and variables common to all objects of the same class.
* Class variables for classes derived from AbstractDbObject are kept in an
* instance of {@link DbObjectClassVariables}.
*
* @return the class variables
*/
public DbObjectClassVariables
getClassVariables() {
throw new PersistenceException(this, "classvariables not initialized for " + getClass());
}
/**
* Gets the basename of the class of this object.
* The basename is the class name without the package name.
*
* @return the basename of the Objects class
*/
public String getClassBaseName () {
return getClassVariables().classBaseName;
}
/**
* Gets the unique class id.
*
* @return the class id
*/
public int getClassId() {
return classId != 0 ? classId : getClassVariables().classId;
}
/**
* Sets the class id for this po.
*
* @param classId the class id, 0 if use id from classvariables
*/
public void setClassId(int classId) {
this.classId = classId;
}
/**
* Clones an object.
* Cloning an object yields a copy with id=0 and serial=0.
* Needs to be overridden if object references, etc... have to be cloned too.
* Subclasses should throw a {@link PersistenceException} if they should not be cloned
* depending on the application logic (not CloneNotSupportedException).
*
* @return a clone of this object
*/
@Override
@SuppressWarnings("unchecked")
public P clone() {
P obj;
try {
obj = (P) super.clone();
}
catch (CloneNotSupportedException ex) {
throw new InternalError(); // should never happen
}
// don't use setters bec. obj might be immutable
((AbstractDbObject) obj).id = 0;
((AbstractDbObject) obj).serial = 0;
return obj;
}
/**
* Updates the attributes in snapshot object.
* The snapshot object is assumed to be a clone of this object.
*
* @param snapshot the snapshot
*/
protected void createAttributesInSnapshot(AbstractDbObject snapshot) {
snapshot.id = id; // was cleared in clone()
snapshot.serial = serial;
}
/**
* Copies all attributes from a snapshot object back to this object.
*
* @param snapshot the snapshot object
*/
protected void revertAttributesToSnapshot(AbstractDbObject snapshot) {
// Important: no snapshot
cause of inheritance problems
id = snapshot.id;
serial = snapshot.serial;
tableSerial = snapshot.tableSerial;
modified = snapshot.modified;
immutable = snapshot.immutable;
preparePending = snapshot.preparePending;
session = snapshot.session;
sessionImmutable = snapshot.sessionImmutable;
sessionHolder = snapshot.sessionHolder;
overloadable = snapshot.overloadable;
modlog = snapshot.modlog;
changeSupport = snapshot.changeSupport;
}
/**
* Accepts a persistence operation visitor.
*
* @param visitor the visitor
* @param modType the modification type
*/
public void acceptPersistenceVisitor(PersistenceVisitor visitor, char modType) {
try {
visitor.visit(this, modType);
}
catch (NoSuchMethodException nsm) {
throw new PersistenceException(getSession(), "no visit method for " + getClass().getName() + " in " + visitor.getClass().getName(), nsm);
}
}
/**
* Adds a PropertyChangeListener to the listener list.
* The listener is registered for all bound properties of this class.
*
* Please notice that all listeners are automatically removed after setModified(false)!
*
* @param listener the property change listener to be added
*
* @see #removePropertyChangeListener
* @see #getPropertyChangeListeners
* @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
* @see #setModified(boolean)
* @see #removeAllPropertyChangeListeners()
*/
public void addPropertyChangeListener(PropertyChangeListener listener) {
if (listener != null) {
synchronized(this) {
if (changeSupport == null) {
changeSupport = new PropertyChangeSupport(this);
}
changeSupport.addPropertyChangeListener(listener);
}
}
}
/**
* Removes a PropertyChangeListener from the listener list.
*
* @param listener the PropertyChangeListener to be removed
*
* @see #addPropertyChangeListener
* @see #getPropertyChangeListeners
* @see #removePropertyChangeListener(java.lang.String,java.beans.PropertyChangeListener)
*/
public void removePropertyChangeListener(PropertyChangeListener listener) {
if (listener != null && changeSupport != null) {
changeSupport.removePropertyChangeListener(listener);
}
}
/**
* Returns an array of all the property change listeners
* registered on this PDO.
*
* @return all PropertyChangeListener
s
* or an empty array if no property change
* listeners are currently registered
*
* @see #addPropertyChangeListener
* @see #removePropertyChangeListener
* @see #getPropertyChangeListeners(java.lang.String)
* @see java.beans.PropertyChangeSupport#getPropertyChangeListeners
*/
public PropertyChangeListener[] getPropertyChangeListeners() {
synchronized (this) {
if (changeSupport == null) {
return new PropertyChangeListener[0];
}
return changeSupport.getPropertyChangeListeners();
}
}
/**
* Adds a PropertyChangeListener to the listener list for a specific
* property.
*
* Note that if this pdo is inheriting a bound property, then no
* event will be fired in response to a change in the inherited property.
*
* If propertyName
or listener
is null
,
* no exception is thrown and no action is taken.
*
* Please notice that all listeners are automatically removed after setModified(false)!
*
* @param propertyName one of the property names listed above
* @param listener the property change listener to be added
*
* @see #removePropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
* @see #getPropertyChangeListeners(java.lang.String)
* @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
* @see #setModified(boolean)
* @see #removeAllPropertyChangeListeners()
*/
public void addPropertyChangeListener(String propertyName, PropertyChangeListener listener) {
if (listener != null) {
synchronized (this) {
if (changeSupport == null) {
changeSupport = new PropertyChangeSupport(this);
}
changeSupport.addPropertyChangeListener(propertyName, listener);
}
}
}
/**
* Removes a PropertyChangeListener
from the listener
* list for a specific property. This method should be used to remove
* PropertyChangeListener
s
* that were registered for a specific bound property.
*
* If propertyName
or listener
is null
,
* no exception is thrown and no action is taken.
*
* @param propertyName a valid property name
* @param listener the PropertyChangeListener to be removed
*
* @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
* @see #getPropertyChangeListeners(java.lang.String)
* @see #removePropertyChangeListener(java.beans.PropertyChangeListener)
*/
public void removePropertyChangeListener(String propertyName, PropertyChangeListener listener) {
if (listener != null && changeSupport != null) {
changeSupport.removePropertyChangeListener(propertyName, listener);
}
}
/**
* Returns an array of all the listeners which have been associated
* with the named property.
*
* @param propertyName a valid property name
* @return all of the PropertyChangeListener
s associated with
* the named property; if no such listeners have been added or
* if propertyName
is null
, an empty
* array is returned
*
* @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
* @see #removePropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
* @see #getPropertyChangeListeners
*/
public PropertyChangeListener[] getPropertyChangeListeners(String propertyName) {
synchronized (this) {
if (changeSupport == null) {
return new PropertyChangeListener[0];
}
return changeSupport.getPropertyChangeListeners(propertyName);
}
}
/**
* Removes all PropertyChangeListeners
.
*
* Using PropertyChangeListeners
implies the risk to "forget"
* about such listeners for long living objects as it is the case in
* desktop client applications. This method safely removes all listeners.
* It is invoked from setModified(false).
*
* @see #setModified(boolean)
*/
public void removeAllPropertyChangeListeners() {
changeSupport = null;
}
/**
* Support for reporting bound property changes for Object properties.
* This method can be called when a bound property has changed and it will
* send the appropriate PropertyChangeEvent to any registered
* PropertyChangeListeners.
*
* @param propertyName the property whose value has changed
* @param oldValue the property's previous value
* @param newValue the property's new value
*/
protected void firePropertyChange(String propertyName, Object oldValue, Object newValue) {
if (changeSupport != null && !Objects.equals(oldValue, newValue)) {
changeSupport.firePropertyChange(propertyName, oldValue, newValue);
}
}
/**
* Sets a (transient) reference to a modification log.
* Used to keep the original user and timestamp when an object gets
* replayed in another server instance.
*
* @param modlog the modification log, null to clear
*/
public void setModificationLog(ModificationLog modlog) {
this.modlog = modlog;
}
/**
* Gets the modification log if object is being replayed.
*
* @return the modlog, null if none (not replayed)
*/
public ModificationLog getModificationLog() {
return modlog;
}
/**
* Defines whether object may be loaded more than once from storage.
* By default, overloading is not allowed.
*
* @param overloadable true if overloading is allowed
*/
public void setOverloadable(boolean overloadable) {
this.overloadable = overloadable;
}
/**
* Gets the overloadable flag.
*
* @return true if overloading is allowed (default is false)
*/
public boolean isOverloadable() {
return overloadable;
}
/**
* Returns whether instances of this class exist as database entities.
* The default is true. An example of a non-entity object is PartialDbObject.
*
* @return true if entity
*/
public boolean isEntity() {
return true;
}
/**
* Creates a new object of the same class.
* The new object belongs to the same persistence connection.
*
* @return the new object
*/
@SuppressWarnings("unchecked")
public P newInstance() {
try {
AbstractDbObject obj = getClass().newInstance();
obj.session = session;
obj.sessionHolder = sessionHolder;
return (P) obj;
}
catch (InstantiationException | IllegalAccessException e) {
throw new PersistenceException(this, "creating new object failed", e);
}
}
/**
* Sets the session holder for this object.
*
* If a holder is set, getSession() will return the session from the holder.
*
* @param sessionHolder the session holder
*/
public void setSessionHolder(SessionHolder sessionHolder) {
this.sessionHolder = sessionHolder;
}
/**
* Gets the session holder.
*
* @return the holder, null if none
*/
public SessionHolder getSessionHolder() {
return sessionHolder;
}
/**
* Sets the db to immutable.
*
* @param sessionImmutable true if db cannot be changed anymore
*/
@Override
public void setSessionImmutable(boolean sessionImmutable) {
this.sessionImmutable = sessionImmutable;
}
/**
* Returns whether the db is immutable.
*
* @return true if immutable
*/
@Override
public boolean isSessionImmutable() {
SessionHolder holder = getSessionHolder();
if (holder != null) {
return holder.isSessionImmutable();
}
return sessionImmutable;
}
/**
* Sets the logical db connection for this object.
*
* @param session the db session
*/
@Override
public void setSession(Session session) {
SessionHolder holder = getSessionHolder();
if (holder != null) {
holder.setSession(session);
}
else {
if (isSessionImmutable() && this.session != session) {
throw new PersistenceException(this.session, "illegal attempt to change the immutable Db of " + this +
" from " + this.session + " to " + session);
}
this.session = (Db) session;
}
}
/**
* Get the logical db connection for this object.
*
* @return the db connection
*/
@Override
public Db getSession() {
SessionHolder holder = getSessionHolder();
if (holder != null) {
return (Db) holder.getSession();
}
return session;
}
/**
* Gets the backend.
*
* @return the backend
*/
public Backend getBackend() {
return getSession().getBackend();
}
/**
* Sets the unique ID of this object.
* Does not set this object to be modified, see {@link #isModified}.
*
* @param id the object id
*/
@Override
public void setId (long id) {
assertMutable();
this.id = id;
}
/**
* Gets the object ID.
* If the object is deleted (negated ID) the returned
* ID is still positive!
*
* @return the object id
*/
@Override
public long getId () {
return id < 0 ? -id : id;
}
/**
* Sets the serial number (modification count).
* Does not set this object to be modified, see {@link #isModified}.
*
* @param serial the serial number
*/
@Override
public void setSerial (long serial) {
assertMutable();
this.serial = serial;
}
/**
* Gets the serial number.
*
* @return the serial number.
*/
@Override
public long getSerial () {
return serial;
}
/**
* Sets the table serial number (table modification count).
* Does not set this object to be modified, see {@link #isModified}.
*
* @param tableSerial the new table serial
*/
public void setTableSerial (long tableSerial) {
assertMutable();
this.tableSerial = tableSerial;
}
/**
* Gets the table serial.
*
* @return the table serial
*/
public long getTableSerial () {
return tableSerial;
}
/**
* Obtains a new ID for this object.
* If the object already has an ID or is deleted (negative ID)
* the ID will _not_ change.
*/
public void newId() {
assertNotRemote();
if (id == 0) {
assertMutable();
id = getIdSource().nextId(getSession());
}
}
/**
* Reserves an ID.
* Reserved IDs are negative.
* A new object with a reserved ID can be distinguished from
* a deleted object by its serial. See also {@link #isVirgin}.
* If the object already has an ID or is deleted (negative ID)
* the ID will _not_ change.
*/
public void reserveId() {
if (id == 0) {
if (getSession().isRemote()) {
try {
id = getRemoteDelegate().obtainReservedId();
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
newId();
id = -id;
}
}
}
/**
* Checks whether this object is already persistant in the db
* or only residing in memory.
* If an object isNew(), it means that it can be inserted.
* This does not mean, that the object never has been stored in the db,
* i.e. it is possible that the object just has been deleted.
*
* @return true if object is not in database, i.e. new (or deleted)
*/
public boolean isNew() {
// deleted objects get the id negated but keep their serial, i.e.
// it's not sufficient to check only the serial.
return id <= 0;
}
/**
* Checks whether the object has a valid ID, i.e. not 0.
*
* @return true if object got a valid id, whether deleted or not
*/
public boolean isIdValid() {
return id != 0;
}
/**
* Checks whether object is deleted.
*
* @return true if object has been deleted
*/
public boolean isDeleted() {
return id < 0 && serial > 0;
}
/**
* Checks whether this object ever was stored in the database.
* Virgin objects have a serial of zero.
* Notice that an object is still "virgin", if it got a valid id via reserveId()
* but has not been saved so far.
*
* @return true if object is virgin, false if it is or was stored in the database.
*/
@Override
public boolean isVirgin() {
return serial == 0;
}
/**
* Sets the modified flag.
* For optimizations it is possible to skip objects that have not
* been modified. The modified-attribute is cleared whenever the
* object is saved (inserted or updated -- NOT insertPlain and updatePlain!).
* The application is responsible to set the modified flag!
* This is usually done in the setter-methods of the db-attributes.
* The wurblet 'DbMethods' does this for you with option '--tracked'
*
* If invoked with modified=false (which is the case after been loaded
* from storage or after successful delete/update/insert) all property
* change listeners will also be removed.
*
* @param modified is true if object is flagged modified, false if not.
*/
public void setModified(boolean modified) {
assertMutable();
this.modified = modified;
if (!modified) {
// loaded from storage or persisted: remove property change listeners
removeAllPropertyChangeListeners();
}
}
/**
* Defines whether attributes of the object may be changed.
*
* Any attempt to invoke a setter on an immutable object
* results in a {@link PersistenceException}.
* All generated methods check this. Note that even lazy loading of
* an immutable object isn't allowed as well because the
* reference to the lazily loaded object must be changed.
*
* @param immutable true if object is immutable
*/
@Override
public void setImmutable(boolean immutable) {
if (immutable && attributesModified()) {
throw new PersistenceException(this, new ImmutableException("object is already modified"));
}
this.immutable = immutable;
}
/**
* Returns whether object is immutable.
*
* By default objects are mutable.
*
* @return true if immutable
*/
@Override
public boolean isImmutable() {
return immutable;
}
@Override
public void setImmutableLoggingLevel(Level immutableLoggingLevel) {
this.immutableLoggingLevel = immutableLoggingLevel;
}
@Override
public Level getImmutableLoggingLevel() {
return immutableLoggingLevel;
}
/**
* Checks if modification of this object are tracked, i.e. setModified()
* is called properly. By default, DbObjects are *NOT* tracked!
* This is quality measure to ensure that isModified() returns
* false if and only if it hasn't been modified, i.e. the setters
* check for modification. See the wurblet DbMethods.
*
* @return true if tracked, false otherwise (default)
*/
public boolean isTracked() {
return false;
}
/**
* Determines whether the object should be written to persistant
* storage because it has been modified.
* By definition, an object is 'modified' if the object OR ANY
* of its components are modified. See DbRelations.wrbl on how
* and when to override isModified().
* New objects are modified by definition!
* Furthermore, isModified() will invoke the errorhandler if
* isTracked() != true. DbMethods automatically override this
* method if option --tracked is given.
*
* @return true if object is modified and should be saved().
*/
public boolean isModified() {
if (!isTracked()) {
throw new PersistenceException(this, "isModified() invoked on untracked object");
}
return attributesModified() || isNew(); // new objects are modified by definition
}
/**
* Determines whether this object got some of its attributes modified.
* It does not check whether some of its components are modified!
* This method can also be used for non-tracked entities.
*
* @return true if this object
*/
public boolean attributesModified() {
return modified;
}
/**
* Returns whether any of the attributes differs from the values persisted in the database.
* This method is only applicable to fulltracked entities and returns false if not fulltracked.
*
* @return true if differs, false if no change or entity isn't fulltracked
*/
public boolean differsPersisted() {
return false;
}
/**
* Determines whether this object is allowed to be stored in DB.
* By default all mutable objects are allowed to be saved.
* Objects not allowed to be saved will force saveObject(), insert() and update()
* to return 'false' and silently skipped in saveCollection().
*
* @return true if saveable
*/
public boolean isPersistable() {
return !isImmutable();
}
/**
* Gets a prepared statement.
*
* @param stmtId the statement id
* @param resultSetType the result set type
* @param resultSetConcurrency the result set concurrency
* @param sqlsupplier the sql code supplier
* @return the statement
*/
public PreparedStatementWrapper getPreparedStatement(
StatementId stmtId, int resultSetType, int resultSetConcurrency, Supplier sqlsupplier) {
return getSession().getPreparedStatement(new StatementKey(stmtId, getClass()), isStatementAlwaysPrepared(),
resultSetType, resultSetConcurrency, sqlsupplier);
}
/**
* Gets a prepared statement.
* Uses {@link java.sql.ResultSet#TYPE_FORWARD_ONLY} and {@link java.sql.ResultSet#CONCUR_READ_ONLY}.
*
* @param stmtId the statement id
* @param sqlsupplier the sql code supplier
* @return the statement
*/
public PreparedStatementWrapper getPreparedStatement(StatementId stmtId, Supplier sqlsupplier) {
return getSession().getPreparedStatement(new StatementKey(stmtId, getClass()), isStatementAlwaysPrepared(), sqlsupplier);
}
/**
* Creates a one-shot prepared statement.
*
* @param sql the sql code
* @param resultSetType the result set type
* @param resultSetConcurrency the result set concurrency
* @return the statement
*/
public PreparedStatementWrapper createPreparedStatement(String sql, int resultSetType, int resultSetConcurrency) {
return getSession().createPreparedStatement(sql, resultSetType, resultSetConcurrency);
}
/**
* Creates a one-shot prepared statement.
* Uses {@link java.sql.ResultSet#TYPE_FORWARD_ONLY} and {@link java.sql.ResultSet#CONCUR_READ_ONLY}.
*
* @param sql the sql code
* @return the statement
*/
public PreparedStatementWrapper createPreparedStatement(String sql) {
return getSession().createPreparedStatement(sql);
}
/**
* Saves all composite relations that reference this object.
*
* @param update true if this is an update operation, else insert
*/
public void saveReferencingRelations(boolean update) {
// default does nothing
}
/**
* Saves all composite relations referenced by this object.
*
* @param update true if this is an update operation, else insert
*/
public void saveReferencedRelations(boolean update) {
// default does nothing
}
/**
* Deletes all composite relations that reference this object.
*/
public void deleteReferencingRelations() {
// default does nothing
}
/**
* Deletes all composite relations referenced by this object.
*/
public void deleteReferencedRelations() {
// default does nothing
}
/**
* Loads lazy references.
*
* The method is used to load any lazy references (not composite) before an object
* is being transferred to another tier. Some lazy references may be necessary
* to persist the object on the other side (due to some weird business logic or
* replicated environments such as PoolKeeper). If those referenced objects are
* not available yet on the remote side (because of an inappropriately ordered stream),
* they must be preloaded on the local side before Serialization.
* Not to mention that those references must not be transient!
*
* The default implementation does nothing.
*/
public void loadLazyReferences() {
// default does nothing
}
/**
* Checks whether this object is referenced by other objects.
*
* It is invoked before operations that may have an impact on the
* referential integrity.
* The default implementation returns false.
*
* The application can assume a lazy context (invoked from is...Lazy)
* if invoked outside a transaction. This is just an optimization hint.
*
* @return true if referenced
*/
public boolean isReferenced() {
if (getSession().isRemote()) {
if (isNew()) {
// new objects are never referenced because they simply don't exist in the db!
// so we can saveObject a roundtrip here
return false;
}
else {
try {
return getRemoteDelegate().isReferenced(id);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
}
else {
// local
return getClassVariables().isReferenced(getSession(), getId());
}
}
/**
* Checks whether this object can be removed.
*
* It is invoked before operations that may have an impact on the
* referential integrity.
* The default implementation returns true if !isNew and !isReferenced.
* Does not refer to the SecurityManager!
*
* The application can assume a lazy context (invoked from is...Lazy)
* if invoked outside a transaction. This is just an optimization hint.
*
* Notice that isRemovable() for performance reasons is not covered
* by its own delegate method in remote connections.
* Hence, if classes in your application require a different implementation
* (!isNew && !isReferenced) you must provide such a remote method.
*
* @return true if removable
*/
public boolean isRemovable() {
return !isNew() && !isReferenced();
}
/**
* Determines whether modifications of this object are counted in the modification table.
* Counting is also turned on if the PDO provides a tableserial.
*
* @param modType is one of the modtypes INSERT, UPDATE, DELETE, ...
* @return true if count modification, false if not.
*/
public boolean isCountingModification(char modType) {
return isTableSerialProvided();
}
/**
* By default objects don't need to include the tableSerial in the
* database table.
* Override this method if object contains a TABLESERIAL-column.
*
* @return true if object is using the tableSerial column, false if not.
*/
public boolean isTableSerialProvided() {
return false;
}
/**
* Determines whether each modification of this object (with respect to the database)
* should be logged.
*
* @param modType is one of the modtypes INSERT, UPDATE, DELETE, ...
* @return true if log modification, false if not.
* @see ModificationLog
*/
public boolean isLoggingModification(char modType) {
return false;
}
/**
* Indicates whether some other object is "equal to" this one.
*
* DbObjects are identical if their IDs and classes are identical.
* IDs are *NOT* necessarily unique among all db-tables! (see poolkeeper).
* Does not throw any exception, so its safe to add null-objects
* to set of DbObjects.
* If this or the passed object has an Id of 0, there is no domain identity
* and {@link Object#equals(java.lang.Object)}
* will be invoked.
*
* @param object the object to test for equality
*/
@Override
public boolean equals (Object object) {
try {
long objectId = ((AbstractDbObject>) object).getId();
if (objectId == 0 || getId() == 0) {
// no Id -> no PDO identity -> use object address
return super.equals(object);
}
return objectId == getId() && getClass() == object.getClass();
}
catch (Exception ex) {
return false;
}
}
/**
* Compare two objects.
* We are using the ID to compare objects. Makes only sense within
* the same class hierarchy.
* Must be overridden if other sortings are desired.
* Does not throw any exception, so its safe to add null-objects
* to a sorted set of DbObjects.
*
* @param obj the object to compare this object with
* @return 0 if objects are equal, < 0 if this is logically less than obj, > 0 if this is logically greater than obj
*/
@Override
public int compareTo(P obj) {
try {
int rv = Long.compare(getId(), obj.getId());
if (rv == 0) {
// should not happen, but one never knows: check the class
if (getClass() != obj.getClass()) { // != is okay here
rv = getClass().getName().compareTo(obj.getClass().getName());
}
}
return rv;
}
catch (Exception ex) {
return 1;
}
}
/**
* The hashcode for a Db-object is simply the ID.
* It is ok -- according to the contract of hashCode() -- that objects
* in different tables with the same id may return the same hashcode.
* If the id is 0, i.e. the object has no domain-identity,
* so {@link Object#hashCode()} is returned.
*
* @return a hash code value for this object.
* @see java.lang.Object#equals(java.lang.Object)
* @see java.util.Hashtable
*/
@Override
public int hashCode() {
if (getId() == 0) {
// no identity: use object's address
return super.hashCode();
}
return (int) getId();
}
/**
* Get the IdSource.
*
* @return the id source
*/
public IdSource getIdSource() {
return getClassVariables().getIdSource(getSession());
}
/**
* Reads the values from a result-set into this object.
*
* @param rs is the result set (wrapper)
* @return the persistent object, never null
*/
@SuppressWarnings("unchecked")
public P readFromResultSetWrapper(ResultSetWrapper rs) {
assertNotOverloaded();
getSession().setAlive(true); // keep the (local) db alive for long running retrievals
try {
getFields(rs);
setModified(false);
return (P) this;
}
catch (PersistenceException de) {
throw de; // no cleanup necessary: done in DbRuntimeException
}
catch (RuntimeException re) {
if (!getSession().isTxRunning()) {
// no transaction running: application cannot invoke rollback -> force cleanup
getSession().forceDetached();
}
throw re;
}
}
/**
* Loads an object from the database by its unique ID.
* For local db connections the current object's attributes will be
* replaced by the database values (i.e. this object is returned).
* For remote connections, a copy of the object in the server is returned.
* Hence, applications should always create a new object and invoke
* select and don't make any further assumptions. This applies to
* all select methods returning an object! Example:
*
* Customer customer = new Customer(db).select(customerId);
*
*
* @param id is the object id
*
* @return object if loaded, null if no such object
*/
@SuppressWarnings("unchecked")
public P selectObject (long id) {
P obj = null;
if (id > 0) {
if (getSession().isRemote()) {
try {
obj = getRemoteDelegate().selectObject(id);
if (obj != null) {
obj.setSession(getSession());
}
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectObjectStatementId,
() -> createSelectSql(false));
st.setLong(1, id);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
obj = readFromResultSetWrapper(rs);
}
}
}
}
return obj;
}
/**
* Load the object from the database with exclusive lock (aka write lock).
* This is implemented via "SELECT FOR UPDATE".
*
* @param id is the object id
*
* @return object if loaded, null if no such object
*/
@SuppressWarnings("unchecked")
public P selectLockedObject (long id) {
P obj = null;
if (id > 0) {
if (getSession().isRemote()) {
try {
obj = getRemoteDelegate().selectLockedObject(id);
if (obj != null) {
obj.setSession(getSession());
}
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
/**
* else local.
*
* Notice that some dbms don't provide a select for update. In this case,
* the dummyUpdate()-method should be automatically used by tentackle
* (allthough it's not really the same, but better than nothing).
* We did not implement it so far, because all tentackle-supported dbms
* provide a SELECT FOR UPDATE.
*
* However, in Oracle (version 8 at the time of writing) the SELECT FOR UPDATE
* is broken under the following circumstances:
* If an object is selected without "FOR UPDATE" and IMMEDIATELY after that
* a transaction is started AND the same object is read with
* SELECT FOR UPDATE within that transaction, a "fetch out of sequence"
* ORA-01002 exception is raised.
* Furthermore, if a transaction that contains a SELECT FOR UPDATE for a given
* object and the same object is selected IMMEDIATELY after the commit
* of the transaction, the db-connection hangs!
* @todo: verify whether that's still the case with newer versions of oracle
*/
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectLockedStatementId,
() -> createSelectSql(true));
st.setLong(1, id);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
obj = readFromResultSetWrapper(rs);
}
}
}
}
return obj;
}
/**
* Reloads the object.
* Note: to make sure that any lazy inits are cleared,
* the returned object is always a new object.
*
* @return the object if reloaded, else null (never this)
*/
public P reloadObject() {
return newInstance().selectObject(id);
}
/**
* Reloads the object with a write lock.
* Note: to make sure that any lazy inits are cleared,
* the returned object is always a new object.
*
* @return the object if reloaded, else null (never this)
*/
public P reloadLockedObject() {
return newInstance().selectLockedObject(id);
}
/**
* Selects all objects of this class and returns the ResultSetWrapper.
*
* @return the result set
*/
public ResultSetWrapper resultAllObjects() {
getSession().assertNotRemote();
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectAllObjectsStatementId,
() -> {
StringBuilder sql = createSelectAllInnerSql();
getSession().getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
return st.executeQuery();
}
/**
* Selects all id,serial-pairs of this class and returns the ResultSetWrapper.
*
* @return the result set
*/
public ResultSetWrapper resultAllIdSerial() {
getSession().assertNotRemote();
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectAllIdSerialStatementId,
() -> {
StringBuilder sql = createSelectAllIdSerialInnerSql();
getSession().getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
return st.executeQuery();
}
/**
* Selects the next object from a resultset.
* AbstractApplications should close the resultset if null is returned.
*
* @param rs the result set
* @return the next object, null if end of set
*/
@SuppressWarnings("unchecked")
public P selectNextObject(ResultSetWrapper rs) {
getSession().assertNotRemote();
if (rs.next()) {
return readFromResultSetWrapper(rs);
}
return null;
}
/**
* Selects all objects of this class as a {@link java.util.List}.
*
* @return the list of objects
*/
@SuppressWarnings("unchecked")
public List extends P> selectAllObjects() {
if (getSession().isRemote()) {
try {
List extends P> list = getRemoteDelegate().selectAllObjects();
for (P obj : list) {
obj.setSession(getSession());
}
return list;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
List
list = new ArrayList<>();
try (ResultSetWrapper rs = resultAllObjects()) {
P obj;
while ((obj = newInstance().selectNextObject(rs)) != null) {
list.add(obj);
}
return list;
}
}
}
/**
* Selects all id,serial-pairs of this class as a list of {@link IdSerialTuple}.
*
* @return the list of objects
*/
public List selectAllIdSerial() {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().selectAllIdSerial();
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
List list = new ArrayList<>();
try (ResultSetWrapper rs = resultAllIdSerial()) {
while (rs.next()) {
IdSerialTuple idSerial = new IdSerialTuple(rs.getLong(1), rs.getLong(2));
list.add(idSerial);
}
return list;
}
}
}
/**
* Selects the serial-number for a given object id.
*
* @param id the object id
* @return the serial for that id, -1 if no such object
*/
public long selectSerial(long id) {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().selectSerial(id);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectSerialStatementId,
this::createSelectSerialSql);
st.setLong(1, id);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
return rs.getLong(1);
}
else {
return -1;
}
}
}
}
/**
* Selects the highest id.
*
* @return the highest id, -1 if table is empty
*/
public long selectMaxId() {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().selectMaxId();
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectMaxIdStatementId,
this::createSelectMaxIdSql);
try (ResultSetWrapper rs = st.executeQuery()) {
long maxId = -1;
if (rs.next()) {
Long val = rs.getALong(1);
if (val != null) {
maxId = val;
}
}
return maxId;
}
}
}
/**
* Selects the highest table serial.
*
* @return the highest table serial, -1 if table is empty
*/
public long selectMaxTableSerial() {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().selectMaxTableSerial();
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectMaxTableSerialStatementId,
this::createSelectMaxTableSerialSql);
try (ResultSetWrapper rs = st.executeQuery()) {
long maxTableSerial = -1;
if (rs.next()) {
Long val = rs.getALong(1);
if (val != null) {
maxTableSerial = val;
}
}
return maxTableSerial;
}
}
}
/**
* Copies an object from one db connection to another.
*
* @param destDb the destination db
* @param plain is true to use insertPlain (recommended), else insert
*/
public void copyToDb(Db destDb, boolean plain) {
Db oldDb = getSession();
setSession(destDb);
try {
if (plain) {
insertPlain();
}
else {
insertObject();
}
}
finally {
setSession(oldDb);
}
}
/**
* Gets the number of columns for this entity class.
* The method does a dummy select if not known so far.
*
* @return the number of columns
*/
public int getColumnCount() {
assertNotRemote();
DbObjectClassVariables classVariables = getClassVariables();
if (classVariables.columnCount <= 0) {
// perform a dummy select (this happens only once per class, hence no prepared statement)
String sql = getBackend().optimizeSql(Backend.SQL_SELECT + createSelectAllInnerSql() + SQL_AND + "1=0");
try (ResultSetWrapper rs = getSession().createStatement().executeQuery(sql)) {
classVariables.columnCount = rs.getColumnCount();
}
}
return classVariables.columnCount;
}
/**
* Insert this object into the database without any further processing
* (i.e. prepareSetFields, linked objcets, mod counting, logging, etc...).
*/
@SuppressWarnings("unchecked")
public void insertPlain() {
if (getSession().isRemote()) {
try {
getRemoteDelegate().insertPlain((P) this);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.INSERT)) {
insertImpl(getClassVariables(), this::createInsertSql);
}
}
}
/**
* Insert implementation.
*
* Should not be used by applications. Will be overridden for multi-inheritance.
*
* @param classVariables the classvariables
* @param sqlsupplier the SQL code supplier
*/
protected void insertImpl(DbObjectClassVariables
classVariables, Supplier sqlsupplier) {
PreparedStatementWrapper st = getPreparedStatement(classVariables.insertStatementId, sqlsupplier);
setFields(st);
assertThisRowAffected(st.executeUpdate());
}
/**
* Deletes this object from the database without any further processing
* (i.e. linked objcets, mod counting, logging, etc...).
*/
public void deletePlain() {
if (getSession().isRemote()) {
try {
getRemoteDelegate().deletePlain(id, serial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.DELETE)) {
deleteImpl(getClassVariables(), this::createDeleteSql);
}
}
}
/**
* Delete implementation.
*
* Should not be used by applications. Will be overridden for multi-inheritance.
*
* @param classVariables the classvariables
* @param sqlsupplier the SQL code supplier
*/
protected void deleteImpl(DbObjectClassVariables
classVariables, Supplier sqlsupplier) {
PreparedStatementWrapper st = getPreparedStatement(classVariables.deleteStatementId, sqlsupplier);
st.setLong(1, id);
st.setLong(2, serial);
assertThisRowAffected(st.executeUpdate());
}
/**
* Updates this object to the database without any further processing
* (i.e. prepareSetFields, linked objcets, mod counting, logging, etc...).
*/
@SuppressWarnings("unchecked")
public void updatePlain () {
if (getSession().isRemote()) {
try {
getRemoteDelegate().updatePlain((P) this);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.UPDATE)) {
updateImpl(getClassVariables(), this::createUpdateSql);
serial++;
}
}
}
/**
* Update implementation.
*
* Should not be used by applications. Will be overridden for multi-inheritance.
*
* @param classVariables the classvariables
* @param sqlsupplier the SQL code supplier
*/
protected void updateImpl(DbObjectClassVariables
classVariables, Supplier sqlsupplier) {
PreparedStatementWrapper st = getPreparedStatement(classVariables.updateStatementId, sqlsupplier);
setFields(st);
assertThisRowAffected(st.executeUpdate());
}
/**
* Performs a dummy update.
* The method is provided as an alternative to {@link #reloadLockedObject} or {@link #selectLockedObject}
* to lock the object during a transaction by updating the ID without changing it.
*/
@SuppressWarnings("unchecked")
public void dummyUpdate () {
if (getSession().isRemote()) {
try {
getRemoteDelegate().dummyUpdate((P) this);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// for Oracle notes see selectLocked().
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().dummyUpdateStatementId,
this::createDummyUpdateSql);
st.setLong(1, id);
assertThisRowAffected(st.executeUpdate());
}
}
/**
* Updates and increments the serial number of this object.
* The method is provided to update an object with isModified() == true
* and 'modified' == false, i.e. an object that itself is not modified
* but some of its components. In such a case it is not necessary
* to update the whole object. However, it is sometimes necessary to
* update the serial to indicate 'some modification' and to make
* sure that this object is part of the transaction.
* Whether it is necessary or not depends on the application.
*
* @see #isUpdatingSerialEvenIfNotModified
*/
public void updateSerial () {
if (getSession().isRemote()) {
try {
getRemoteDelegate().updateSerial(id, serial);
serial++;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.UPDATE)) {
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateSerialStatementId,
this::createUpdateSerialSql);
st.setLong(1, id);
st.setLong(2, serial);
assertThisRowAffected(st.executeUpdate());
serial++; // SQL statement incremented serial successfully
}
}
}
/**
* Same as {@link #updateSerial} but updates tableSerial as well.
* Notice: the tableSerial is NOT modified in the current object,
* but only in the database!
*/
public void updateSerialAndTableSerial () {
if (getSession().isRemote()) {
try {
getRemoteDelegate().updateSerialAndTableSerial(id, serial, tableSerial);
serial++;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.UPDATE)) {
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateSerialAndTableSerialStatementId,
this::createUpdateSerialAndTableSerialSql);
st.setLong(1, tableSerial);
st.setLong(2, id);
st.setLong(3, serial);
assertThisRowAffected(st.executeUpdate());
serial++; // SQL statement incremented serial successfully
}
}
}
/**
* Determines whether in updates of composite objects unmodified objects in the
* update path get at least the serial updated or are not touched at all.
* The default is to leave unmodified objects untouched.
* However, in some applications it is necessary to update the master object if some of its childs are updated (usually
* to trigger something, e.g. a cache-update).
* The default implementation returns false.
* Override this method to change to 'true'.
*
* @return true if update serial even if object is unchanged
* @see #updateObject
*/
public boolean isUpdatingSerialEvenIfNotModified() {
return false;
}
/**
* Does any preprocessing before delete, insert or update.
*
* @param modType is the modification type: DELETE, INSERT or UPDATE
*/
public void initModification(char modType) {
if (isCountingModification(modType)) {
setTableSerial(countModification());
// will never return on failure -> errorHandler
}
}
/**
* Does any postprocessing after delete, insert or update.
*
* @param modType is the modification type: DELETE, INSERT or UPDATE
*/
public void finishModification(char modType) {
if (isLoggingModification(modType)) {
logModification(modType);
}
}
/**
* Does any update postprocessing for objects not being updated (because not modified
* for some reason, e.g. only components modified).
* The default implementation does nothing. See {@link org.tentackle.pdo.AbstractPersistentObject}.
*
* @param modType is the modification type: DELETE, INSERT or UPDATE
*/
public void finishNotUpdated(char modType) {
}
/**
* Begins a transaction.
* Also sets the txObject if a transaction was started.
*
* @param txName the transaction name
* @return true if tx begun
*/
protected long beginTx(String txName) {
long txVoucher = getSession().begin(txName);
if (txVoucher != 0) {
getSession().setTxObject(this);
}
return txVoucher;
}
/**
* Inserts this (new) object into the database.
* Note: this method does *NOT* set the ID and should be used
* by the application with great care! Use {@link #saveObject} instead!
*/
public void insertObject() {
assertPersistable();
prepareSetFields();
if (getSession().isRemote()) {
try {
@SuppressWarnings("unchecked")
DbObjectResult result = getRemoteDelegate().insertObject((P) this);
id = result.getId();
serial = result.getSerial();
tableSerial = result.getTableSerial();
setModified(false);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.INSERT)) {
long oldId = id; // remember old id and serial
long oldSerial = serial;
long txVoucher = beginTx(TX_INSERT_OBJECT);
try {
initModification(ModificationLog.INSERT);
if (id < 0) {
id = -id; // could have been "deleted" before
}
serial++;
saveReferencedRelations(false);
insertImpl(getClassVariables(), this::createInsertSql);
saveReferencingRelations(false);
finishModification(ModificationLog.INSERT);
getSession().commit(txVoucher);
setModified(false);
}
catch (RuntimeException ex) {
// application has thrown an exception
// if tx was begun, the tx will be rolled back and all pending
// statements (marked ready) will be consumed.
// If no rollback, the exception must be caught elsewhere in the path
// until a valid rollback can be performed.
getSession().rollback(txVoucher); // rollback if tx was begun
serial = oldSerial;
id = oldId;
if (ex instanceof PersistenceException) {
((PersistenceException) ex).updateDbObject(this);
throw ex;
}
throw new PersistenceException(this, ex);
}
}
}
}
/**
* Returns whether update of this object is necessary.
*
* @return true if update necessary
*/
protected boolean isUpdateNecessary() {
return !isTracked() || attributesModified() || isNew();
}
/**
* Updates this object to the database.
* The modified attribute gets cleared if insert was successful.
* Note: this method should be used by the application with great care!
* Use {@link #saveObject} instead!
*/
public void updateObject() {
assertPersistable();
prepareSetFields();
if (getSession().isRemote()) {
try {
@SuppressWarnings("unchecked")
DbObjectResult result = getRemoteDelegate().updateObject((P) this);
id = result.getId();
serial = result.getSerial();
tableSerial = result.getTableSerial();
setModified(false);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.UPDATE)) {
long oldId = id; // remember old id and serial
long oldSerial = serial;
long txVoucher = beginTx(TX_UPDATE_OBJECT); // start transaction
try {
boolean updated = true; // cleared to false if this object was NOT updated, only components
if (!isUpdateNecessary()) { // DON'T USE isModified() here because its generated/overridden
// the object itself is not modified, but some of its components
if (isUpdatingSerialEvenIfNotModified()) {
initModification(ModificationLog.UPDATE);
// DON'T USE updateSerial() here! We need the old serial unchanged
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateSerialStatementId,
() -> isTableSerialProvided() ? createUpdateSerialAndTableSerialSql() : createUpdateSerialSql());
saveReferencedRelations(true);
// id can't be negative cause if isNew() above
int ndx = 0;
if (isTableSerialProvided()) {
st.setLong(++ndx, tableSerial);
}
st.setLong(++ndx, id);
st.setLong(++ndx, serial);
assertThisRowAffected(st.executeUpdate());
saveReferencingRelations(true);
finishModification(ModificationLog.UPDATE);
}
else {
// saveObject linked objects without updating the serial of this object
saveReferencedRelations(true);
saveReferencingRelations(true);
finishNotUpdated(ModificationLog.UPDATE);
updated = false; // this object was not updated! -> no modlog as well, no serial++!
}
}
else {
// normal update
initModification(ModificationLog.UPDATE);
if (id < 0) {
id = -id; // was deleted: reuse ID
}
saveReferencedRelations(true);
updateImpl(getClassVariables(), this::createUpdateSql);
saveReferencingRelations(true);
finishModification(ModificationLog.UPDATE);
}
getSession().commit(txVoucher);
if (updated) {
serial++; // serial is already incremented in the SQL-statement!
}
setModified(false); // clear modified flag
}
catch (RuntimeException ex) {
// application has thrown an exception
// if tx was begun, the tx will be rolled back and all pending
// statements (marked ready) will be consumed.
// If no rollback, the exception must be caught elsewhere in the path
// until a valid rollback can be performed.
getSession().rollback(txVoucher); // rollback if tx was begung
id = oldId;
serial = oldSerial;
if (ex instanceof PersistenceException) {
((PersistenceException) ex).updateDbObject(this);
throw ex;
}
throw new PersistenceException(this, ex);
}
}
}
}
/**
* Prepares this object for saveObject.
* The method is invoked at the client-side only, i.e. 3-tier or 2-tier, but never at the server-side in 3-tier mode.
* The default implementation clears a non-transient flag that tells the server that prepareSave() already has been
* invoked on the client side. AbstractApplications may override the method to perform any modifications to the object
* before it is sent to the server (instead of {@link #prepareSetFields()} which is invoked at the server-side only).
*
* Example:
*
* public void prepareSave() {
* // do something
* ...
* super.prepareSave(); // IMPORTANT!!!
* }
*
Notice that the method is invoked outside of the transaction.
*/
public void prepareSave() {
preparePending = false;
}
/**
* Clears some references to reduce bandwidth.
* Invoked for remote objects only.
*/
public void clearOnRemoteSave() {
// default does nothing
}
/**
* Saves this object.
* This is the standard method applications should use to insert or update
* objects.
* If the ID is 0, a new ID is obtained and this object inserted.
* Otherwise this object is updated.
* The modified attribute gets cleared if save() was successful.
*/
public void saveObject () {
preparePending = true;
if (getSession().isRemote()) {
// execute in 3-tier client
clearOnRemoteSave();
prepareSave();
try {
@SuppressWarnings("unchecked")
DbObjectResult result = getRemoteDelegate().saveObject((P) this);
id = result.getId();
serial = result.getSerial();
tableSerial = result.getTableSerial();
setModified(false);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
if (preparePending) {
// execute in 2-tier client
prepareSave();
}
long oldId = id; // saveObject in case of error
long txVoucher = beginTx(TX_SAVE); // obtaining ID should be used in transaction with insert()
try {
if (id <= 0) {
// object is new or contains a reserved ID (e.g. is deleted)
if (id == 0) {
// object is new: obtain new id
newId();
}
// if id has been reserved it will be made positive in insert or update
// dont't use insertObject() bec. insert() OR insertObject()
insertObject();
}
else {
// object already exists in database
updateObject();
}
getSession().commit(txVoucher);
}
catch (RuntimeException ex) {
getSession().rollback(txVoucher);
id = oldId; // set back old Id in case it was changed
if (ex instanceof PersistenceException) {
((PersistenceException) ex).updateDbObject(this);
throw ex;
}
throw new PersistenceException(this, ex);
}
}
}
/**
* Persists this object and returns it.
*
* @return the persistet object, null if saveObject failed
*/
@SuppressWarnings("unchecked")
public P persistObject() {
if (getSession().isRemote()) {
// execute in 3-tier client
clearOnRemoteSave();
preparePending = true;
prepareSave();
try {
P obj = getRemoteDelegate().persistObject((P) this);
if (obj != null) {
obj.setSession(getSession());
}
return obj; // obj != this !!!
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
saveObject();
return (P) this;
}
}
/**
* Prepares this object for delete.
* The method is invoked at the client-side only, i.e. 3-tier or
* 2-tier, but never at the server-side in 3-tier mode.
* The default implementation clears a non-transient flag that tells
* the server that prepareDelete() already has been invoked on the client side.
* AbstractApplications may override the method to perform any modifications to the
* object before it is sent to the server.
*
* Example:
*
* public void prepareDelete() {
* // do something
* ...
* super.prepareDelete(); // IMPORTANT!!!
* }
*
* Notice that the method is invoked outside of the transaction and that it
* is not invoked if the object is new or not saveable.
*/
public void prepareDelete() {
preparePending = false;
}
/**
* Removes this object from the database (with linked objects, i.e. standard-case).
* A removed object will also get the modified attribute set by definition,
* because it {@link #isNew} again.
* It is also verified that the object {@link #isPersistable}.
*
* @throws NotFoundException if object is new or not saveable or not in database
*/
public void deleteObject() {
assertNotNew();
assertPersistable();
preparePending = true;
if (getSession().isRemote()) {
prepareDelete();
try {
@SuppressWarnings("unchecked")
DbObjectResult result = getRemoteDelegate().deleteObject((P) this);
id = result.getId();
serial = result.getSerial();
tableSerial = result.getTableSerial();
setModified(false);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
if (preparePending) {
prepareDelete();
}
if (getSession().isPersistenceOperationAllowed(this, ModificationLog.DELETE)) {
long txVoucher = beginTx(TX_DELETE_OBJECT);
try {
initModification(ModificationLog.DELETE);
deleteReferencingRelations();
deleteImpl(getClassVariables(), this::createDeleteSql);
deleteReferencedRelations();
finishModification(ModificationLog.DELETE);
getSession().commit(txVoucher);
id = -id; // make ID reserved again, i.e. mark object as being deleted
setModified(false);
}
catch (RuntimeException ex) {
// application has thrown an exception
// if tx was begun, the tx will be rolled back and all pending
// statements (marked ready) will be consumed.
// If no rollback, the exception must be caught elsewhere in the path
// until a valid rollback can be performed.
getSession().rollback(txVoucher); // rollback if tx was begun
if (ex instanceof PersistenceException) {
((PersistenceException) ex).updateDbObject(this);
throw ex;
}
throw new PersistenceException(this, ex);
}
}
}
}
/**
* Marks an object to be deleted.
* This is done by negating its id.
* If the object is already marked deleted the method does nothing.
* Must be overridden if the object is composite, i.e.
* all its components must be markDeleted as well.
* Note: an object with a negative ID is always {@link #isModified}.
* @see #unmarkDeleted()
*/
public void markDeleted() {
// getId() always returns >= 0!
setId(-getId());
}
/**
* Removes the deleted mark.
* @see #markDeleted()
*/
public void unmarkDeleted() {
setId(getId());
}
/**
* Counts a modification for the class of this object.
*
* @return the table serial,
* -1 if isCountModificationAllowed() == false
*/
public long countModification () {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().countModification();
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
return getSession().isCountModificationAllowed() ?
PdoTracker.getInstance().countModification(getTableName()) : -1;
}
}
/**
* Selects the current modification counter for the class of this object.
*
* @return the modification counter
*/
public long getModificationCount () {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().getModificationCount();
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
return ((ModificationTracker) PdoTracker.getInstance()).getIdSerialForName(getTableName()).getSerial();
}
}
/**
* Determines the objects with a tableSerial starting at a given serial.
* Useful to cleanup caches for example.
*
* @param oldSerial non-inclusive lower bound for tableSerial (> oldSerial)
* @return pairs of longs, the first being the ID, the second the tableserial, never null
*/
public List selectExpiredTableSerials(long oldSerial) {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().selectExpiredTableSerials(oldSerial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectExpiredTableSerials1StatementId,
this::createSelectExpiredTableSerials1Sql);
st.setLong(1, oldSerial);
try (ResultSetWrapper rs = st.executeQuery()) {
List expireList = new ArrayList<>();
while (rs.next()) {
expireList.add(new IdSerialTuple(rs.getLong(1), rs.getLong(2)));
}
return expireList;
}
}
}
/**
* Determines the objects with their tableSerial within a given range.
* Useful to cleanup caches.
*
* @param oldSerial non-inclusive lower bound for tableSerial (> oldSerial)
* @param maxSerial inclusive upper bound for tableSerial (≤ maxSerial)
* @return pairs of longs, the first being the ID, the second the tableserial, never null
*/
public List selectExpiredTableSerials(long oldSerial, long maxSerial) {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().selectExpiredTableSerials(oldSerial, maxSerial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectExpiredTableSerials2StatementId,
this::createSelectExpiredTableSerials2Sql);
st.setLong(1, oldSerial);
st.setLong(2, maxSerial);
List expireList = new ArrayList<>();
try (ResultSetWrapper rs = st.executeQuery()) {
while (rs.next()) {
expireList.add(new IdSerialTuple(rs.getLong(1), rs.getLong(2)));
}
return expireList;
}
}
}
/**
* Gets the expiration backlog for a given range of tableserials.
* Note that the backlog is maintained only if DbGlobal.serverDb != null.
*
* @param minSerial the lower serial bound of the query (minSerial < tableSerial)
* @param maxSerial the upper serial bound of the query (tableSerial ≤ maxSerial)
* @return the expiration info as pairs of id/tableserial, null if given range was not found in the backlog
*/
public List getExpirationBacklog(long minSerial, long maxSerial) {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().getExpirationBacklog(minSerial, maxSerial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
// local
return getClassVariables().expirationBacklog.getExpiration(minSerial, maxSerial);
}
}
/**
* Combines {@link #selectExpiredTableSerials} and {@link #getExpirationBacklog}.
* A physical database query is only done if the requested range is not in the backlog.
* Used in RMIservers to reduces db-roundtrips.
*
* @param oldSerial non-inclusive lower bound for tableSerial (> oldSerial)
* @param maxSerial inclusive upper bound for tableSerial (≤ maxSerial)
* @return pairs of longs, the first being the ID, the second the tableserial, never null
*/
public List getExpiredTableSerials(long oldSerial, long maxSerial) {
if (getSession().isRemote()) {
try {
return getRemoteDelegate().getExpiredTableSerials(oldSerial, maxSerial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
else {
List exp;
if (AbstractApplication.getRunningApplication().isServer()) {
final TableSerialExpirationBacklog backlog = getClassVariables().expirationBacklog;
synchronized(backlog) {
// try to get from backlog
exp = backlog.getExpiration(oldSerial, maxSerial);
if (exp == null) {
// no info in backlog: physically select
exp = selectExpiredTableSerials(oldSerial, maxSerial);
backlog.addExpiration(oldSerial, maxSerial, exp);
LOGGER.fine("added expiration set {0}-{1}[{2}] for {3}",
oldSerial, maxSerial, exp.size(), getTableName());
}
}
}
else {
exp = selectExpiredTableSerials(oldSerial, maxSerial);
}
return exp;
}
}
/**
* Creates a {@link ModificationLog}.
*
* @param modType is the modification type
* @return the created ModificationLog ready for saveObject()
*/
public ModificationLog createModificationLog(char modType) {
return ModificationLogFactory.getInstance().createModificationLog(this, modType);
}
/**
* Logs a modification for this object to the modlog
* (not to be mixed up with the modification-counter!)
*
* @param modType is the modification type (INSERT, DELETE or UPDATE)
*/
public void logModification(char modType) {
getSession().assertNotRemote(); // don't log locally if db is remote
if (getSession().isLogModificationAllowed()) {
getSession().logBeginTx(); // optionally create the begin log
createModificationLog(modType).saveObject();
}
}
/**
* Gets the delegate for remote connections.
* Each class has its own delegate.
*
* @return the delegate for this object
*/
public AbstractDbObjectRemoteDelegate getRemoteDelegate() {
return getClassVariables().getRemoteDelegate(getSession());
}
/**
* Determines whether prepared statements of this class should always
* be prepared each time when the statement used.
* @return true if always prepare
*/
public boolean isStatementAlwaysPrepared() {
return getClassVariables().alwaysPrepare;
}
/**
* Sets the always prepare flag.
*
* @param alwaysPrepare true if always prepare
*/
public void setStatementAlwaysPrepared(boolean alwaysPrepare) {
getClassVariables().alwaysPrepare = alwaysPrepare;
}
/**
* Gets the database table name for the class of this object.
* @return the table name
*/
public String getTableName() {
return getClassVariables().tableName;
}
/**
* Retrieves the values of all fields.
*
* @param rs the result set
*/
public void getFields(ResultSetWrapper rs) {
// default does nothing
}
/**
* Prepares the object's attributes before the object is saved to the database.
* The default implementation does nothing.
* Used to setup, check and align values.
*/
public void prepareSetFields() {
}
/**
* Sets the values of all fields (all columns of the database table)
* in the given {@link PreparedStatementWrapper} from the object's attributes.
*
* @param st the statement
* @return the number of fields set
*/
public int setFields (PreparedStatementWrapper st) {
return 0; // no fields processed so far
}
/**
* Creates the inner sql text to select all fields.
*
* Returns something like:
*
* "* FROM xytable WHERE 1=1"
*
*
* @return the sql text
*/
public StringBuilder createSelectAllInnerSql() {
StringBuilder sql = new StringBuilder();
sql.append(SQL_ALLSTAR);
sql.append(SQL_FROM);
sql.append(getTableName());
sql.append(SQL_WHEREALL);
return sql;
}
/**
* Creates the inner sql text to select the id and serial fields.
*
* Returns something like:
*
* "id,serial FROM xytable WHERE 1=1"
*
*
* @return the sql text
*/
public StringBuilder createSelectAllIdSerialInnerSql() {
StringBuilder sql = new StringBuilder();
sql.append(CN_ID);
sql.append(SQL_COMMA);
sql.append(CN_SERIAL);
sql.append(SQL_FROM);
sql.append(getTableName());
sql.append(SQL_WHEREALL);
return sql;
}
/**
* Creates the inner sql text to select all fields by ID.
*
* Returns something like:
*
* "* FROM xytable WHERE id=?"
*
*
* @return the sql text
*/
public StringBuilder createSelectAllByIdInnerSql() {
StringBuilder sql = createSelectAllInnerSql();
sql.append(SQL_AND);
sql.append(CN_ID);
sql.append(SQL_EQUAL_PAR);
return sql;
}
/**
* Creates the inner sql text to select the ID field.
*
* Returns something like:
*
* "id FROM xytable WHERE 1=1"
*
*
* @return the sql text
*/
public StringBuilder createSelectIdInnerSql() {
StringBuilder sql = new StringBuilder();
sql.append(CN_ID);
sql.append(SQL_FROM);
sql.append(getTableName());
sql.append(SQL_WHEREALL);
return sql;
}
/**
* Creates the sql text to delete objects.
*
* Returns something like:
*
* "DELETE FROM xytable WHERE 1=1"
*
*
* @return the sql text
*/
public StringBuilder createDeleteAllSql() {
StringBuilder sql = new StringBuilder(SQL_DELETE);
sql.append(SQL_FROM);
sql.append(getTableName());
sql.append(SQL_WHEREALL);
return sql;
}
/**
* Creates the sql intro text to update objects.
*
* Returns something like:
*
* "UPDATE xytable SET "
*
*
* @return the sql text
*/
public StringBuilder createSqlUpdate() {
StringBuilder sql = new StringBuilder(SQL_UPDATE);
sql.append(getTableName());
sql.append(SQL_SET);
return sql;
}
/**
* Creates the SQL code for the update statement.
*
* @return the SQL code
*/
public String createUpdateSql() {
throw new PersistenceException(this, "method createUpdateSql not implemented in " + getClass());
}
/**
* Creates the SQL code for the insert statement.
*
* @return the SQL code
*/
public String createInsertSql() {
throw new PersistenceException(this, "method createInsertSql not implemented in " + getClass());
}
/**
* Creates the SQL code for the select by id statement.
*
* @param locked true if select locked (FOR UPDATE)
* @return the sql code
*/
public String createSelectSql(boolean locked) {
StringBuilder sql = createSelectAllByIdInnerSql();
getSession().getBackend().buildSelectSql(sql, locked, 0, 0);
return sql.toString();
}
/**
* Creates the SQL code for the selectSerial statement.
*
* @return the sql code
*/
public String createSelectSerialSql() {
return SQL_SELECT + CN_SERIAL + SQL_FROM + getTableName() + SQL_WHERE + CN_ID + SQL_EQUAL_PAR;
}
/**
* Creates the SQL code for the selectMaxId statement.
*
* @return the sql code
*/
public String createSelectMaxIdSql() {
return SQL_SELECT + "MAX(" + CN_ID + ")" + SQL_FROM + getTableName();
}
/**
* Creates the SQL code for the selectMaxTableSerial statement.
*
* @return the sql code
*/
public String createSelectMaxTableSerialSql() {
return SQL_SELECT + "MAX(" + CN_TABLESERIAL + ")" + SQL_FROM + getTableName();
}
/**
* Creates the SQL code for the delete statement.
*
* @return the sql code
*/
public String createDeleteSql() {
return SQL_DELETE + SQL_FROM + getTableName() + SQL_WHERE + CN_ID + SQL_EQUAL_PAR + SQL_AND + CN_SERIAL + SQL_EQUAL_PAR;
}
/**
* Creates the SQL code for the dummy update statement.
* Useful get an exclusive lock within a transaction.
*
* @return the sql code
*/
public String createDummyUpdateSql() {
return SQL_UPDATE + getTableName() + SQL_SET + CN_ID + SQL_EQUAL + CN_ID + SQL_WHERE + CN_ID + SQL_EQUAL_PAR;
}
/**
* Creates the SQL code for the serial update statement.
*
* @return the sql code
*/
public String createUpdateSerialSql() {
return SQL_UPDATE + getTableName() + SQL_SET + CN_SERIAL + "=" + CN_SERIAL + "+1" + SQL_WHERE +
CN_ID + SQL_EQUAL_PAR + SQL_AND + CN_SERIAL + SQL_EQUAL_PAR;
}
/**
* Creates the SQL code for the serial + tableSerial update statement.
*
* @return the sql code
*/
public String createUpdateSerialAndTableSerialSql() {
return SQL_UPDATE + getTableName() + SQL_SET + CN_SERIAL + "=" + CN_SERIAL + "+1, " +
CN_TABLESERIAL + SQL_EQUAL_PAR + SQL_WHERE +
CN_ID + SQL_EQUAL_PAR + SQL_AND + CN_SERIAL + SQL_EQUAL_PAR;
}
/**
* Creates the SQL code for the first statement to select expired table serials.
*
* @return the sql code
*/
public String createSelectExpiredTableSerials1Sql() {
// sort by tableserial+id to return pairs in deterministic order for same tableserials
return SQL_SELECT + CN_ID + SQL_COMMA + CN_TABLESERIAL + SQL_FROM + getTableName() + SQL_WHERE +
CN_TABLESERIAL + ">?" + SQL_ORDERBY + CN_TABLESERIAL + SQL_COMMA + CN_ID;
}
/**
* Creates the SQL code for the second statement to select expired table serials.
*
* @return the sql code
*/
public String createSelectExpiredTableSerials2Sql() {
// sort by tableserial+id to return pairs in deterministic order for same tableserials
return SQL_SELECT + CN_ID + SQL_COMMA + CN_TABLESERIAL + SQL_FROM + getTableName() + SQL_WHERE +
CN_TABLESERIAL + ">?" + SQL_AND + CN_TABLESERIAL + "<=?" +
SQL_ORDERBY + CN_TABLESERIAL + SQL_COMMA + CN_ID;
}
/**
* asserts that this object does not belong to a remote session.
*/
protected void assertNotRemote() {
if (getSession().isRemote()) {
throw new PersistenceException(this, "operation not allowed for objects belonging to a remote session");
}
}
/**
* asserts that this object belongs to a remote session.
*/
protected void assertRemote() {
if (!getSession().isRemote()) {
throw new PersistenceException(this, "operation only allowed for objects belonging to a remote session");
}
}
/**
* Checks the correct number of rows affected.
*
* Use whenever an executeUpdate is not related to this object.
*
* @param count the effective number of rows affected
* @param expected the expected number of rows affected
*
* @throws NotFoundException if count < 1
*/
protected void assertNumberOfRowsAffected(int count, int expected) {
if (count != expected) {
String message = "unexpected number of rows affected: " + count + ", expected: " + expected;
if (count < 1) {
throw new NotFoundException(getSession(), message);
}
throw new PersistenceException(getSession(), message);
}
}
/**
* Checks that exactly one row is affected.
*
* Use whenever an executeUpdate is related to this object.
*
* @param count the effective number of rows affected
*/
protected void assertThisRowAffected(int count) {
if (count != 1) {
if (count < 1) {
String message = "no rows affected";
if (!isVirgin()) {
// read the serial to figure out whether object exists
long persistedSerial = selectSerial(getId());
if (persistedSerial > 0) {
// object exists, but probably with wrong serial (depends on the statement)
message += " (persisted serial=" + persistedSerial + ")";
}
else {
message += " (object with id=" + id + " doesn't exist in table " + getTableName() + ")";
}
}
throw new NotFoundException(this, message);
}
else {
throw new PersistenceException(this, "more than one row affected: " + count);
}
}
}
/**
* Asserts that this object is saveable.
*/
protected void assertPersistable() {
if (!isPersistable()) {
throw new PersistenceException(this, "object is not persistable");
}
}
/**
* Asserts that object is not new.
*/
protected void assertNotNew() {
if (isNew()) {
throw new NotFoundException(this, id == 0 ? "object is new" : "object is deleted");
}
}
/**
* Asserts that this object is not overloaded.
*/
protected void assertNotOverloaded() {
if (!isOverloadable() && !isVirgin()) {
// overloading disabled and object is not virgin:
throw new PersistenceException(this, "object is already loaded");
}
}
/**
* Asserts that object is mutable.
*/
protected void assertMutable() {
if (isImmutable()) {
PersistenceException ex = new PersistenceException(this, new ImmutableException("object is immutable"));
if (immutableLoggingLevel == null) {
throw ex;
}
LOGGER.log(immutableLoggingLevel, ex.getMessage(), ex);
}
}
}