All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.tentackle.dbms.AbstractDbObject Maven / Gradle / Ivy

The newest version!
/*
 * Tentackle - https://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.dbms;

import org.tentackle.dbms.rmi.AbstractDbObjectRemoteDelegate;
import org.tentackle.dbms.rmi.DbObjectResult;
import org.tentackle.log.Logger;
import org.tentackle.log.Logger.Level;
import org.tentackle.misc.IdSerialTuple;
import org.tentackle.misc.Immutable;
import org.tentackle.misc.ImmutableException;
import org.tentackle.misc.PropertyListener;
import org.tentackle.misc.PropertySupport;
import org.tentackle.session.ModificationTracker;
import org.tentackle.session.NotFoundException;
import org.tentackle.session.PersistenceException;
import org.tentackle.session.Session;
import org.tentackle.session.SessionHolder;
import org.tentackle.sql.Backend;

import java.io.Serial;
import java.lang.reflect.InvocationTargetException;
import java.rmi.RemoteException;
import java.util.ArrayList;
import java.util.List;

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 persistence implementations must extend {@code AbstractDbObject}, which provides the generic * functionality of persistent objects. Every {@code 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 usually generated * by wurblets (see wurbelizer.org) according to a model. * * @param

the persistent class type */ public abstract class AbstractDbObject

> implements Immutable, ModificationLoggable, Comparable

{ /** * Instantiates a new db object for a given class without a session. * * @param

the class type * @param clazz the class * @return the object */ public static

> P newInstance(Class

clazz) { try { // load the class return clazz.getDeclaredConstructor().newInstance(); } catch (InstantiationException | IllegalAccessException | InvocationTargetException | NoSuchMethodException 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 */ public static

> 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; /** 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"; private static final Logger LOGGER = Logger.get(AbstractDbObject.class); @Serial private static final long serialVersionUID = 1L; // data object attributes are always private private long id; // unique Object ID private long serial; // serial-nummer (version to detect simultaneous updates) private int classId; // the class id (overrides classvariables classid) 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 boolean finallyImmutable; // true if object is finally immutable private Level immutableLoggingLevel; // optional logging for immutable violations instead of exception // 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 final transient boolean fromThisJVM = true; // true if object created in this JVM, else remote private transient boolean overloadable; // true if object can be loaded more than once from the database private transient volatile PropertySupport propertySupport; // for PropertyListeners /** * Creates a database object. * * @param db the session */ public AbstractDbObject(Db db) { setSession(db); } /** * Creates a database object not associated to a session.
* The session 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 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 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 to use id from class-variables */ public void setClassId(int classId) { this.classId = classId; } /** * Returns whether this object has been created in this JVM. * * @return true if this JVM, else from remote JVM */ public boolean isFromThisJVM() { return fromThisJVM; } public boolean isCopy() { return false; // implemented in AbstractPO! } // there is no createAttributesInSnapshot, because all attributes are either immutable or frozen. /** * Copies all attributes from a snapshot object back to this object. *

* There is no createAttributesInSnapshot, since the snapshot is cloned and all attributes are either immutable or frozen. * * @param snapshot the snapshot object */ @SuppressWarnings("rawtypes") protected void revertAttributesToSnapshot(AbstractDbObject snapshot) { // id, serial, tableSerial are not reverted because they can only be changed by a persistence operation if (!isCopy()) { // copies are always mutable and new modified = snapshot.modified; immutable = snapshot.immutable; finallyImmutable = snapshot.finallyImmutable; overloadable = snapshot.overloadable; propertySupport = snapshot.propertySupport; sessionImmutable = snapshot.sessionImmutable; } session = snapshot.session; sessionHolder = snapshot.sessionHolder; } /** * Accepts a persistence operation visitor.
* * @param visitor the visitor * @param modType the modification type */ public void acceptPersistenceVisitor(PersistenceVisitor visitor, ModificationType modType) { try { visitor.visit(this, new Class[] {Character.TYPE}, modType); } catch (NoSuchMethodException nsm) { throw new PersistenceException(getSession(), "no visit method for " + getClass().getName() + " in " + visitor.getClass().getName(), nsm); } } /** * Creates a property support object for this persistent object. * * @return the support object */ protected PropertySupport createPropertySupport() { return new PropertySupport(this); } /** * Gets the property support.
* Creates it atomically if it doesn't exist. * * @return the property support, never null */ protected PropertySupport getPropertySupport() { PropertySupport localSupport = propertySupport; if (localSupport == null) { synchronized (this) { localSupport = propertySupport; if (localSupport == null) { localSupport = propertySupport = createPropertySupport(); } } } return localSupport; } /** * Adds a {@link PropertyListener} 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 #setModified(boolean) */ public void addPropertyListener(PropertyListener listener) { if (listener != null) { getPropertySupport().addPropertyListener(listener); } } /** * Removes a {@link PropertyListener} from the listener list. * * @param listener the PropertyChangeListener to be removed */ public void removePropertyListener(PropertyListener listener) { if (listener != null) { PropertySupport localSupport = propertySupport; if (localSupport != null) { localSupport.removePropertyListener(listener); } } } /** * Adds a {@link PropertyListener} to the listener list for a specific * property. *

* 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 #setModified(boolean) */ public void addPropertyListener(String propertyName, PropertyListener listener) { if (listener != null) { getPropertySupport().addPropertyListener(propertyName, listener); } } /** * Removes a {@link PropertyListener} from the listener * list for a specific property. This method should be used to remove * PropertyChangeListeners * that were registered for a specific bound property. * * @param propertyName a valid property name * @param listener the PropertyChangeListener to be removed */ public void removePropertyListener(String propertyName, PropertyListener listener) { if (listener != null) { PropertySupport localSupport = propertySupport; if (localSupport != null) { localSupport.removePropertyListener(propertyName, listener); } } } /** * Removes all {@link PropertyListener}s. *

* Using PropertyListeners 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 removeAllPropertyListeners() { propertySupport = 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) { PropertySupport localSupport = propertySupport; if (localSupport != null) { localSupport.firePropertyChanged(propertyName, oldValue, newValue); } } /** * 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 session. * * @return the new object */ @SuppressWarnings({ "unchecked", "rawtypes" }) public P newInstance() { try { AbstractDbObject obj = getClass().getDeclaredConstructor().newInstance(); obj.session = session; obj.sessionHolder = sessionHolder; return (P) obj; } catch (InstantiationException | IllegalAccessException | InvocationTargetException | NoSuchMethodException 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 session for this object. * * @param session the 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 session for this object. * * @return the session */ @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 */ 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 */ 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; } } } /** * Reserves a given ID.
* It doesn't matter whether the ID is negative or positive. * Reserving 0 clears the reservation. *

* Throws a {@link PersistenceException} if the object is not new. * * @param id the ID to reserve, 0 to clear the reservation */ public void reserveId(long id) { assertNew(); this.id = id < 0 ? id : -id; } /** * Checks whether this object is already persistent 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 has never 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 attributes. *

* If invoked with modified=false (which is the case after being 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 removeAllPropertyListeners(); } } /** * 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. Notice that lazy loading * is still allowed. * * @param immutable true if object is immutable */ @Override public void setImmutable(boolean immutable) { if (immutable) { if (attributesModified()) { // wrapped in PersistenceException due to "this" argument throw new PersistenceException(this, new ImmutableException("object is already modified")); } } else if (isFinallyImmutable()) { throw new PersistenceException(this, new ImmutableException("object is finally immutable")); } this.immutable = immutable; } @Override public void setFinallyImmutable() { setImmutable(true); finallyImmutable = true; } /** * Returns whether object is immutable. *

* By default, objects are mutable. * * @return true if immutable */ @Override public boolean isImmutable() { return immutable; } @Override public boolean isFinallyImmutable() { return finallyImmutable; } @Override public void setImmutableLoggingLevel(Level immutableLoggingLevel) { this.immutableLoggingLevel = immutableLoggingLevel; } @Override public Level getImmutableLoggingLevel() { return immutableLoggingLevel; } /** * Checks if modification of this object are tracked.
* By default, AbstractDbObjects 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 persistent * 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 the modified flag.
* Necessary for fulltracked PDOs only, since attributedModified will be overridden with differsPersisted * and setModified(true) invoked by the application would not have any effect. * * @return the modified flag */ public boolean isForcedModified() { 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 savable */ public boolean isPersistable() { return !isImmutable(); } /** * Creates a statement key for the corresponding persistence class from a statement id. * * @param stmtId the statement id * @return the statement key */ protected StatementKey createStatementKey(StatementId stmtId) { return new StatementKey(stmtId, getClass()); } /** * 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, SqlSupplier sqlSupplier) { return getSession().getPreparedStatement(createStatementKey(stmtId), 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, SqlSupplier sqlSupplier) { return getSession().getPreparedStatement(createStatementKey(stmtId), isStatementAlwaysPrepared(), sqlSupplier); } /** * Creates a one-shot prepared statement.
* * @param sqlSupplier the SQL code supplier * @param resultSetType the result set type * @param resultSetConcurrency the result set concurrency * @return the statement */ public PreparedStatementWrapper createPreparedStatement(SqlSupplier sqlSupplier, int resultSetType, int resultSetConcurrency) { return getSession().createPreparedStatement(sqlSupplier, 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 sqlSupplier the SQL code supplier * @return the statement */ public PreparedStatementWrapper createPreparedStatement(SqlSupplier sqlSupplier) { return getSession().createPreparedStatement(sqlSupplier); } /** * 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() { Db db = getSession(); if (db.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(db, 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 the modification type * @return true if count modification, false if not. */ public boolean isCountingModification(ModificationType 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 database modification of this object should be logged. * * @param modType the modification type * @return true if log modification, false if not. * @see ModificationLog */ public boolean isLoggingModification(ModificationType modType) { return false; } /** * Determines whether the modification of this object should be replayed in lenient mode. * * @param modType the modification type * @return true if leniently * @see ModificationLog */ public boolean isReplayedLeniently(ModificationType modType) { return false; } /** * Indicates whether some other object is "equal to" this one.
* Method is final!
*

* Objects are identical if their IDs and classes are identical.
* IDs are *NOT* necessarily unique among all database tables! * 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 final boolean equals (Object object) { if (object != null && getClass() == object.getClass()) { 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(); } return false; } /** * Compare two objects.
* Objects are compared by ID.
* Method is final!
* * @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 final int compareTo(P obj) { if (obj != null) { int rv = Long.compare(getId(), obj.getId()); if (rv == 0 && // should not happen, but one never knows: check the class getClass() != obj.getClass()) { // != is okay here rv = getClass().getName().compareTo(obj.getClass().getName()); } return rv; } return 1; } /** * The hashcode is the ID.
* Method is final!
* 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 final 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(); Db db = getSession(); db.setAlive(true); // keep the (local) db alive for long-running retrievals try { getFields(rs); setModified(false); return (P) this; } catch (PersistenceException px) { throw px; // no cleanup necessary: done in PersistenceException } catch (RuntimeException re) { if (!db.isTxRunning()) { // no transaction running: application cannot invoke rollback -> force cleanup db.forceDetached(); } throw re; } } /** * Loads an object from the database by its unique ID.

* For local sessions 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 */ public P selectObject (long id) { P obj = null; if (id > 0) { Db db = getSession(); if (db.isRemote()) { try { obj = getRemoteDelegate().selectObject(id); if (obj != null) { obj.setSession(db); } } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectObjectStatementId, b -> createSelectSql(b, 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 */ public P selectObjectForUpdate(long id) { P obj = null; if (id > 0) { Db db = getSession(); if (db.isRemote()) { try { obj = getRemoteDelegate().selectObjectForUpdate(id); if (obj != null) { obj.setSession(db); } } 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 * (although 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 JDBC-connection hangs! * @todo: verify whether that's still the case with newer versions of Oracle */ getSession().assertTxRunning(); PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectForUpdateStatementId, b -> createSelectSql(b, 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. * * @return the object if reloaded, else null (never this) */ public P reloadObjectForUpdate() { return newInstance().selectObjectForUpdate(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, b -> { StringBuilder sql = createSelectAllInnerSql(b); b.buildSelectSql(sql, false, 0, 0); return sql.toString(); } ); return st.executeQuery(); } /** * Selects all objects of this class with a higher tableserial and returns the ResultSetWrapper. * * @param oldSerial the last known serial * @return the result set */ public ResultSetWrapper resultObjectsWithExpiredTableSerials(long oldSerial) { getSession().assertNotRemote(); PreparedStatementWrapper st = getPreparedStatement(getClassVariables().selectObjectsWithExpiredTableSerialsStatementId, b -> { StringBuilder sql = createSelectObjectsWithExpiredTableSerialsSql(b); b.buildSelectSql(sql, false, 0, 0); return sql.toString(); } ); st.setLong(1, oldSerial); 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, b -> { StringBuilder sql = createSelectAllIdSerialInnerSql(); b.buildSelectSql(sql, false, 0, 0); return sql.toString(); } ); return st.executeQuery(); } /** * Selects the next object from a result-set. * AbstractApplications should close the result-set if null is returned. * * @param rs the result set * @return the next object, null if end of set */ 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, never null */ public List

selectAllObjects() { Db db = getSession(); if (db.isRemote()) { try { List

list = getRemoteDelegate().selectAllObjects(); for (P obj : list) { obj.setSession(db); } 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 objects with a tableSerial starting at a given serial.
* Useful to update expired objects in a batch. * * @param oldSerial non-inclusive lower bound for tableSerial (> oldSerial) * @return the list of objects, never null */ public List

selectObjectsWithExpiredTableSerials(long oldSerial) { Db db = getSession(); if (db.isRemote()) { try { List

list = getRemoteDelegate().selectObjectsWithExpiredTableSerials(oldSerial); for (P obj : list) { obj.setSession(db); } return list; } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { // local List

list = new ArrayList<>(); try (ResultSetWrapper rs = resultObjectsWithExpiredTableSerials(oldSerial)) { 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, b -> 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, b -> 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, b -> createSelectMaxTableSerialSql()); try (ResultSetWrapper rs = st.executeQuery()) { long maxTableSerial = -1; if (rs.next()) { Long val = rs.getALong(1); if (val != null) { maxTableSerial = val; } } return maxTableSerial; } } } /** * 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) Backend backend = getBackend(); String sql = backend.optimizeSql(Backend.SQL_SELECT + createSelectAllInnerSql(backend) + SQL_AND + "1=0"); StatementWrapper stmt = getSession().createStatement(); stmt.setParallelOk(true); // don't count this as a parallel query try (ResultSetWrapper rs = stmt.executeQuery(sql)) { classVariables.columnCount = rs.getColumnCount(); } } return classVariables.columnCount; } /** * Insert this object into the database without any further processing * (i.e. prepareSetFields, linked objects, mod counting, logging, etc...). */ @SuppressWarnings("unchecked") public void insertPlain() { Db db = getSession(); if (db.isRemote()) { try { getRemoteDelegate().insertPlain((P) this); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { // local if (db.isPersistenceOperationAllowed(this, DbModificationType.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, SqlSupplier 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 objects, mod counting, logging, etc...). */ public void deletePlain() { Db db = getSession(); if (db.isRemote()) { try { getRemoteDelegate().deletePlain(id, serial); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { // local if (db.isPersistenceOperationAllowed(this, DbModificationType.DELETE)) { deleteImpl(getClassVariables(), b -> 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, SqlSupplier 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 objects, mod counting, logging, etc...). */ @SuppressWarnings("unchecked") public void updatePlain () { Db db = getSession(); if (db.isRemote()) { try { getRemoteDelegate().updatePlain((P) this); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { if (db.isPersistenceOperationAllowed(this, DbModificationType.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, SqlSupplier 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 #reloadObjectForUpdate} or {@link #selectObjectForUpdate} * 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, b -> 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 () { Db db = getSession(); if (db.isRemote()) { try { getRemoteDelegate().updateSerial(id, serial); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { if (db.isPersistenceOperationAllowed(this, DbModificationType.UPDATE)) { PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateSerialStatementId, b -> createUpdateSerialSql()); st.setLong(1, id); st.setLong(2, serial); assertThisRowAffected(st.executeUpdate()); } } serial++; // SQL statement incremented serial successfully } /** * Updates and sets the serial number of this object.
* Caution: this is a low-level method provided to fix the serial * if it is wrong for whatever reason. * * @param serial the new serial */ public void updateSerial(long serial) { Db db = getSession(); if (db.isRemote()) { try { getRemoteDelegate().updateAndSetSerial(id, serial); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { if (db.isPersistenceOperationAllowed(this, DbModificationType.UPDATE)) { PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateAndSetSerialStatementId, b -> createUpdateAndSetSerialSql()); st.setLong(1, serial); st.setLong(2, id); assertThisRowAffected(st.executeUpdate()); } } this.serial = serial; } /** * 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 () { Db db = getSession(); if (db.isRemote()) { try { getRemoteDelegate().updateSerialAndTableSerial(id, serial, tableSerial); serial++; } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { if (db.isPersistenceOperationAllowed(this, DbModificationType.UPDATE)) { PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateSerialAndTableSerialStatementId, b -> 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 children 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 the modification type */ protected void initModification(ModificationType modType) { if (isCountingModification(modType)) { setTableSerial(countModification()); // will never return on failure -> errorHandler } } /** * Does any postprocessing after delete, insert or update. * * @param modType the modification type */ protected void finishModification(ModificationType modType) { if (isLoggingModification(modType)) { logModification(modType); } } /** * Does any update postprocessing for objects not being updated.
* Necessary, because not modified for some reason, e.g. only components modified.
* The default implementation does nothing. * * @param modType the modification type */ protected void finishNotUpdated(ModificationType modType) { } /** * Prepares the attributes concerning composite relations.
* Overridden, for example, to update list counts. */ protected void alignComponents() { // default does nothing } /** * Returns whether update of this object is necessary. * * @return true if update necessary */ protected boolean isUpdateNecessary() { alignComponents(); return !isTracked() || attributesModified() || isNew(); } /** * 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! */ @SuppressWarnings("unchecked") public void insertObject() { assertPersistable(); Db db = getSession(); if (db.isRemote()) { try { applyDbObjectResult(getRemoteDelegate().insertObject((P) this)); setModified(false); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { prepareSetFields(); alignComponents(); if (db.isPersistenceOperationAllowed(this, DbModificationType.INSERT)) { long oldId = id; // remember old id and serial long oldSerial = serial; long txVoucher = db.begin(TX_INSERT_OBJECT); try { initModification(DbModificationType.INSERT); if (id < 0) { id = -id; // could have been "deleted" before } serial++; saveReferencedRelations(false); insertImpl(getClassVariables(), this::createInsertSql); saveReferencingRelations(false); finishModification(DbModificationType.INSERT); db.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. db.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); } } } } /** * 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! */ @SuppressWarnings("unchecked") public void updateObject() { assertPersistable(); Db db = getSession(); if (db.isRemote()) { try { applyDbObjectResult(getRemoteDelegate().updateObject((P) this)); setModified(false); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { prepareSetFields(); if (db.isPersistenceOperationAllowed(this, DbModificationType.UPDATE)) { long oldId = id; // remember old id and serial long oldSerial = serial; long txVoucher = db.begin(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(DbModificationType.UPDATE); // DON'T USE updateSerial() here! We need the old serial unchanged PreparedStatementWrapper st = getPreparedStatement(getClassVariables().updateSerialStatementId, b -> 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(DbModificationType.UPDATE); } else { // saveObject linked objects without updating the serial of this object saveReferencedRelations(true); saveReferencingRelations(true); finishNotUpdated(DbModificationType.UPDATE); updated = false; // this object was not updated! -> no modlog as well, no serial++! } } else { // normal update initModification(DbModificationType.UPDATE); if (id < 0) { id = -id; // was deleted: reuse ID } saveReferencedRelations(true); updateImpl(getClassVariables(), this::createUpdateSql); saveReferencingRelations(true); finishModification(DbModificationType.UPDATE); } db.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. db.rollback(txVoucher); // rollback if tx was begun 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.
* Applications 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 JDBC-side only). *

* Notice that the method is invoked outside the save-transaction. * * @see #prepareSetFields() */ public void prepareSave() { } /** * 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. */ @SuppressWarnings("unchecked") public void saveObject () { Db db = getSession(); if (db.isRemote()) { // execute in 3-tier client clearOnRemoteSave(); prepareSave(); try { applyDbObjectResult(getRemoteDelegate().saveObject((P) this)); setModified(false); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { if (isFromThisJVM()) { // not from remote client prepareSave(); } long oldId = id; // saveObject in case of error long txVoucher = db.begin(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 // don't use insertObject() bec. insert() OR insertObject() insertObject(); } else { // object already exists in database updateObject(); } db.commit(txVoucher); } catch (RuntimeException ex) { db.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 persisted object, null if saveObject failed */ @SuppressWarnings("unchecked") public P persistObject() { Db db = getSession(); if (db.isRemote()) { // execute in 3-tier client clearOnRemoteSave(); prepareSave(); try { P obj = getRemoteDelegate().persistObject((P) this); if (obj != null) { obj.setSession(db); } 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. * Applications may override the method to perform any modifications to the * object before it is sent to the server. *

* Example: *

   * void prepareDelete() {
   *   // do something
   *   ...
   *   super.prepareDelete();   // IMPORTANT!!!
   * }
   * 
* Notice that the method is invoked outside the transaction and that it * is not invoked if the object is new or not savable. * * @see #deleteObject() * @see #isNew() * @see #isPersistable() */ public void prepareDelete() { } /** * Removes this object from the database.
* This includes all components, if any.
* 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} and is not new. * * @throws NotFoundException if object is new or not savable or not in database */ @SuppressWarnings("unchecked") public void deleteObject() { assertNotNew(); assertPersistable(); Db db = getSession(); if (db.isRemote()) { prepareDelete(); try { applyDbObjectResult(getRemoteDelegate().deleteObject((P) this)); setModified(false); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { // local if (isFromThisJVM()) { // not from remote client prepareDelete(); } if (db.isPersistenceOperationAllowed(this, DbModificationType.DELETE)) { long txVoucher = db.begin(TX_DELETE_OBJECT); try { initModification(DbModificationType.DELETE); deleteReferencingRelations(); deleteImpl(getClassVariables(), b -> createDeleteSql()); deleteReferencedRelations(); finishModification(DbModificationType.DELETE); db.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. db.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 () { Db db = getSession(); if (db.isRemote()) { try { return getRemoteDelegate().countModification(); } catch (RemoteException e) { throw PersistenceException.createFromRemoteException(this, e); } } else { return db.isCountModificationAllowed() ? ModificationTracker.getInstance().countModification(db, getTableName()) : -1; } } /** * Selects the current modification counter for the class of this object. * * @return the modification counter */ public long getModificationCount () { return ModificationTracker.getInstance().getSerial(getTableName()); } /** * Determines the objects with a tableSerial starting at a given serial. * Useful to clean up 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, b -> 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 clean up 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, b -> 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 RMI-servers to reduces database 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; 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()); } } return exp; } } /** * Logs a modification for this object to the modlog * (not to be mixed up with the modification-counter!) * * @param modType the modification type */ public void logModification(ModificationType modType) { Db db = getSession(); db.assertNotRemote(); // don't log locally if db is remote if (db.isLogModificationAllowed()) { db.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 set up, check and align values. *

* Notice that this method is invoked at the JDBC-side only, i.e. local sessions, * not in remote clients. * * @see #prepareSave() */ 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"
   * 
* * @param backend the backend * @return the sql text */ public StringBuilder createSelectAllInnerSql(Backend backend) { 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=?"
   * 
* * @param backend the backend * @return the sql text */ public StringBuilder createSelectAllByIdInnerSql(Backend backend) { StringBuilder sql = createSelectAllInnerSql(backend); 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. * * @param backend the backend * @return the SQL code */ public String createUpdateSql(Backend backend) { throw new PersistenceException(this, "method createUpdateSql not implemented in " + getClass()); } /** * Creates the SQL code for the insert statement. * * @param backend the backend * @return the SQL code */ public String createInsertSql(Backend backend) { throw new PersistenceException(this, "method createInsertSql not implemented in " + getClass()); } /** * Creates the SQL code for the select by id statement. * * @param backend the backend * @param locked true if select locked (FOR UPDATE) * @return the sql code */ public String createSelectSql(Backend backend, boolean locked) { StringBuilder sql = createSelectAllByIdInnerSql(backend); backend.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 set and update statement. * * @return the sql code */ public String createUpdateAndSetSerialSql() { return SQL_UPDATE + getTableName() + SQL_SET + CN_SERIAL + SQL_EQUAL_PAR + SQL_WHERE + CN_ID + 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; } /** * Creates the SQL code for the first statement to select expired table serials. * * @param backend the backend * @return the sql code */ public StringBuilder createSelectObjectsWithExpiredTableSerialsSql(Backend backend) { StringBuilder sql = createSelectAllInnerSql(backend); sql.append(SQL_AND). append(CN_TABLESERIAL).append(">?"). append(SQL_ORDERBY).append(CN_TABLESERIAL).append(SQL_COMMA).append(CN_ID); return sql; } /** * Applies a {@link DbObjectResult} to this persistence object. * * @param result the remote result */ protected void applyDbObjectResult(DbObjectResult result) { id = result.id(); serial = result.serial(); tableSerial = result.tableSerial(); } /** * 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"; long persistedSerial = 0; boolean temporary = false; if (!isVirgin()) { // read the serial to figure out whether object exists in database persistedSerial = selectSerial(getId()); if (persistedSerial > 0) { // object exists, but probably with wrong serial (depends on the statement) message += " (persisted serial=" + persistedSerial + ")"; temporary = true; // optimistic locking failure? may disappear if whole tx is retried (see @Transaction) } else { message += " (object with id=" + id + " doesn't exist in table " + getTableName() + ")"; } } PersistenceException nfx = new NotFoundException(this, message, persistedSerial); nfx.setTemporary(temporary); throw nfx; } else { throw new PersistenceException(this, "more than one row affected: " + count); } } } /** * Asserts that this object is savable. */ protected void assertPersistable() { if (!isPersistable()) { throw new PersistenceException( this, (isImmutable() ? "immutable " : "") + "object is not persistable"); } } /** * Asserts that object is not new. */ protected void assertNotNew() { if (isNew()) { throw new PersistenceException(this, id == 0 ? "object is new" : "object is deleted"); } } /** * Asserts that object is new. */ protected void assertNew() { if (!isNew()) { throw new PersistenceException(this, "object is not new"); } } /** * 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); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy