org.tentackle.persist.ModificationLog 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.io.Serializable;
import java.rmi.RemoteException;
import java.text.ParseException;
import java.util.ArrayList;
import java.util.List;
import org.tentackle.common.Timestamp;
import org.tentackle.log.Logger;
import org.tentackle.log.LoggerFactory;
import org.tentackle.misc.Canonicalizer;
import org.tentackle.misc.DateHelper;
import org.tentackle.misc.ParameterString;
import org.tentackle.misc.StringHelper;
import org.tentackle.pdo.Pdo;
import org.tentackle.pdo.PdoUtilities;
import org.tentackle.pdo.PersistenceException;
import org.tentackle.pdo.Persistent;
import org.tentackle.pdo.PersistentDomainObject;
import org.tentackle.pdo.Session;
import org.tentackle.persist.rmi.ModificationLogRemoteDelegate;
import org.tentackle.sql.Backend;
/**
* @> $mapfile
*
* # modification table for async coupling
* name := $classname
* id := $classid
* table := $tablename
*
* [remote]
*
* ## attributes
* long objectId objectid object id
* int objectClassId classid object class id
* long txId txid transaction id (optional)
* String(64) txName txname transaction name (optional) [TRIMWRITE]
* char modType modtype modification type
* Timestamp when modtime time of event
* String(32) user moduser name of user [TRIMWRITE]
* String message message optional informational or error message [NOMETHOD]
* Timestamp processed processed processing time [MAPNULL]
*
* ## indexes
* index next := processed, id
* index txid := txid
* index object := processed, objectid, classid
* index user := processed, moduser
*
* @<
*/
/**
* Logging for object modifications.
*
* Modifications to {@link AbstractDbObject}s can be logged to a so-called modification log (aka: modlog).
* Most applications will use the modlog for asynchroneous database coupling.
*
* Note: the txId is only valid (> 0) if the db-connection has {@link Db#isLogModificationTxEnabled},
* i.e. begin and commit records are logged as well. If the {@link IdSource} of the modlog is
* transaction-based, transactions will not overlap in the modlog because obtaining
* the id for the modlog is part of the transaction. However, if the idsource is
* remote (poolkeeper rmi-client, for example), transactions may overlap!
* In such cases the txid is necessary to separate the modlog sequences into
* discrete transactions. (see the PoolKeeper project)
*/
public class ModificationLog extends AbstractDbObject {
private static final long serialVersionUID = 7997968053729155282L;
/**
* logger for this class.
*/
private static final Logger LOGGER = LoggerFactory.getLogger(ModificationLog.class);
/** the database tablename. */
public static final String TABLENAME = /**/"modlog"/**/; // @wurblet < Inject --string $tablename
private static final DbObjectClassVariables CLASSVARIABLES =
new DbObjectClassVariables<>(ModificationLog.class,
/**/2/**/, // @wurblet < Inject $classid
TABLENAME);
/** modification type: begin transaction **/
public static final char BEGIN = 'B';
/** modification type: commit transaction **/
public static final char COMMIT= 'C';
/** modification type: object inserted **/
public static final char INSERT= 'I';
/** modification type: object updated **/
public static final char UPDATE= 'U';
/** modification type: object deleted **/
public static final char DELETE= 'D';
// @wurblet fieldlenghts ColumnNames --model=$mapfile
// //GEN-BEGIN:fieldlenghts
/** database column name for 'objectId'. */
public static final String CN_OBJECTID = "objectid";
/** database column name for 'objectClassId'. */
public static final String CN_OBJECTCLASSID = "classid";
/** database column name for 'txId'. */
public static final String CN_TXID = "txid";
/** database column name for 'txName'. */
public static final String CN_TXNAME = "txname";
/** database column name for 'modType'. */
public static final String CN_MODTYPE = "modtype";
/** database column name for 'when'. */
public static final String CN_WHEN = "modtime";
/** database column name for 'user'. */
public static final String CN_USER = "moduser";
/** database column name for 'message'. */
public static final String CN_MESSAGE = "message";
/** database column name for 'processed'. */
public static final String CN_PROCESSED = "processed";
// //GEN-END:fieldlenghts
// @wurblet fieldnames ColumnLengths --model=$mapfile
// //GEN-BEGIN:fieldnames
/** maximum number of characters for 'txName'. */
int CL_TXNAME = 64;
/** maximum number of characters for 'user'. */
int CL_USER = 32;
// //GEN-END:fieldnames
/**
* The {@link AbstractDbObject} the log belongs to. null = unknown. Speeds up {@link #getDbObject}
* in distributed applications (see the poolkeeper framework).
*
* If the lazyObject is a PersistentObject, it's getPdo() method will hold the PDO.
*/
protected AbstractDbObject> lazyObject;
/**
* Message parameters (lazy)
*/
protected ParameterString messageParameters;
// @wurblet declare Declare --model=$mapfile
// //GEN-BEGIN:declare
/** object id. */
private long objectId;
/** object class id. */
private int objectClassId;
/** transaction id (optional). */
private long txId;
/** transaction name (optional). */
private String txName;
/** modification type. */
private char modType;
/** time of event. */
private Timestamp when;
/** the snapshot of when. */
private transient Timestamp whenSnapshot;
/** name of user. */
private String user;
/** optional informational or error message. */
private String message;
/** processing time. */
private Timestamp processed;
/** the snapshot of processed. */
private transient Timestamp processedSnapshot;
// //GEN-END:declare
/**
* Creates an empty modification log for a given db.
* Useful for reading the log or as an RMI-proxy.
*
* @param db the database connection
*/
public ModificationLog(Db db) {
super(db);
}
/**
* Creates a modification log for a given db and modification type.
*
* @param db the database connection
* @param modType is the modification type (BEGIN or COMMIT)
*/
public ModificationLog(Db db, char modType) {
this(db);
this.modType = modType;
txName = db.getTxName();
txId = db.getLogModificationTxId();
when = DateHelper.now();
user = db.getSessionInfo().getUserName();
}
/**
* Creates a modification log from an object.
*
* @param
the logged object type
* @param object is the logged object
* @param modType is the modification type (INSERT, UPDATE...)
*/
public
> ModificationLog(AbstractDbObject
object, char modType) {
this(object.getSession(), modType);
if (modType == BEGIN || modType == COMMIT) {
throw new PersistenceException(this, "illegal BEGIN or COMMIT in object logging");
}
objectId = object.getId();
/**
* The modlog's serial should reflect the serial of the object.
* The modlog is inserted _after_ the object has been modified in the db.
* Because the modlog's serial will be incremented during save(), we need to subtract 1
* from the serial. However, during update, the serial will be incremented _after_
* creating the modlog (see AbstractDbObject.updateObject()), so we need to subtract 1 only
* for the other modlog types.
*/
setSerial(object.getSerial() - (modType == UPDATE ? 0 : 1));
objectClassId = object.getClassId();
if (modType == INSERT || modType == UPDATE) {
// keep object for RMI transfers (not DELETE as this will be loaded on the servers side)
lazyObject = object;
}
}
/**
* Creates a modlog from another modlog, but a different type.
*
* @param template the modlog template
* @param modType is the modification type (INSERT, UPDATE...)
*/
public ModificationLog(ModificationLog template, char modType) {
super(template.getSession());
objectId = template.objectId;
objectClassId = template.objectClassId;
txId = template.txId;
txName = template.txName;
when = template.when;
user = template.user;
message = template.message;
this.modType = modType;
}
/**
* Creates an empty modlog.
* Constructor only provided for {@link Class#newInstance}.
*/
public ModificationLog() {
super();
}
// @wurblet methods MethodsImpl --model=$mapfile --noif
// //GEN-BEGIN:methods
@Override
public void getFields(ResultSetWrapper rs) {
super.getFields(rs);
if (rs.configureSection(CLASSVARIABLES)) {
rs.configureColumn(CN_OBJECTID);
rs.configureColumn(CN_OBJECTCLASSID);
rs.configureColumn(CN_TXID);
rs.configureColumn(CN_TXNAME);
rs.configureColumn(CN_MODTYPE);
rs.configureColumn(CN_WHEN);
rs.configureColumn(CN_USER);
rs.configureColumn(CN_MESSAGE);
rs.configureColumn(CN_PROCESSED);
rs.configureColumn(CN_ID);
rs.configureColumn(CN_SERIAL);
}
if (rs.getRow() <= 0) {
throw new PersistenceException(getSession(), "no valid row");
}
objectId = rs.getLong();
objectClassId = rs.getInt();
txId = rs.getLong();
txName = rs.getString();
modType = rs.getChar();
when = rs.getTimestamp();
user = rs.getString();
message = rs.getString();
processed = rs.getTimestamp(true);
setId(rs.getLong());
setSerial(rs.getLong());
}
@Override
public int setFields(PreparedStatementWrapper st) {
int ndx = super.setFields(st);
st.setLong(++ndx, objectId);
st.setInt(++ndx, objectClassId);
st.setLong(++ndx, txId);
st.setString(++ndx, StringHelper.trim(txName, 64));
st.setChar(++ndx, modType);
st.setTimestamp(++ndx, when);
st.setString(++ndx, StringHelper.trim(user, 32));
st.setString(++ndx, message);
st.setTimestamp(++ndx, processed, true);
st.setLong(++ndx, getId());
st.setLong(++ndx, getSerial());
return ndx;
}
@Override
public String createInsertSql() {
return Backend.SQL_INSERT_INTO + getTableName() + Backend.SQL_LEFT_PARENTHESIS +
CN_OBJECTID + Backend.SQL_COMMA +
CN_OBJECTCLASSID + Backend.SQL_COMMA +
CN_TXID + Backend.SQL_COMMA +
CN_TXNAME + Backend.SQL_COMMA +
CN_MODTYPE + Backend.SQL_COMMA +
CN_WHEN + Backend.SQL_COMMA +
CN_USER + Backend.SQL_COMMA +
CN_MESSAGE + Backend.SQL_COMMA +
CN_PROCESSED + Backend.SQL_COMMA +
CN_ID + Backend.SQL_COMMA +
CN_SERIAL +
Backend.SQL_INSERT_VALUES +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR_COMMA +
Backend.SQL_PAR + Backend.SQL_RIGHT_PARENTHESIS;
}
@Override
public String createUpdateSql() {
return Backend.SQL_UPDATE + getTableName() + Backend.SQL_SET +
CN_OBJECTID + Backend.SQL_EQUAL_PAR_COMMA +
CN_OBJECTCLASSID + Backend.SQL_EQUAL_PAR_COMMA +
CN_TXID + Backend.SQL_EQUAL_PAR_COMMA +
CN_TXNAME + Backend.SQL_EQUAL_PAR_COMMA +
CN_MODTYPE + Backend.SQL_EQUAL_PAR_COMMA +
CN_WHEN + Backend.SQL_EQUAL_PAR_COMMA +
CN_USER + Backend.SQL_EQUAL_PAR_COMMA +
CN_MESSAGE + Backend.SQL_EQUAL_PAR_COMMA +
CN_PROCESSED + Backend.SQL_EQUAL_PAR_COMMA +
CN_SERIAL + Backend.SQL_EQUAL + CN_SERIAL + Backend.SQL_PLUS_ONE +
Backend.SQL_WHERE + CN_ID + Backend.SQL_EQUAL_PAR +
Backend.SQL_AND + CN_SERIAL + Backend.SQL_EQUAL_PAR;
}
/**
* Gets the attribute objectId.
*
* @return object id
*/
public long getObjectId() {
return objectId;
}
/**
* Sets the attribute objectId.
*
* @param objectId object id
*/
public void setObjectId(long objectId) {
assertMutable();
this.objectId = objectId;
}
/**
* Gets the attribute objectClassId.
*
* @return object class id
*/
public int getObjectClassId() {
return objectClassId;
}
/**
* Sets the attribute objectClassId.
*
* @param objectClassId object class id
*/
public void setObjectClassId(int objectClassId) {
assertMutable();
this.objectClassId = objectClassId;
}
/**
* Gets the attribute txId.
*
* @return transaction id (optional)
*/
public long getTxId() {
return txId;
}
/**
* Sets the attribute txId.
*
* @param txId transaction id (optional)
*/
public void setTxId(long txId) {
assertMutable();
this.txId = txId;
}
/**
* Gets the attribute txName.
*
* @return transaction name (optional)
*/
public String getTxName() {
return txName;
}
/**
* Sets the attribute txName.
*
* @param txName transaction name (optional)
*/
public void setTxName(String txName) {
assertMutable();
this.txName = txName;
}
/**
* Gets the attribute modType.
*
* @return modification type
*/
public char getModType() {
return modType;
}
/**
* Sets the attribute modType.
*
* @param modType modification type
*/
public void setModType(char modType) {
assertMutable();
this.modType = modType;
}
/**
* Gets the attribute when.
*
* @return time of event
*/
public Timestamp getWhen() {
return when;
}
/**
* Sets the attribute when.
*
* @param when time of event
*/
public void setWhen(Timestamp when) {
assertMutable();
this.when = when;
if (when != null) {
when.setUTC(false);
}
}
/**
* Gets the attribute user.
*
* @return name of user
*/
public String getUser() {
return user;
}
/**
* Sets the attribute user.
*
* @param user name of user
*/
public void setUser(String user) {
assertMutable();
this.user = user;
}
/**
* no accessor methods for message.
* optional informational or error message
*/
/**
* Gets the attribute processed.
*
* @return processing time
*/
public Timestamp getProcessed() {
return processed;
}
/**
* Sets the attribute processed.
*
* @param processed processing time
*/
public void setProcessed(Timestamp processed) {
assertMutable();
this.processed = processed;
if (processed != null) {
processed.setUTC(false);
}
}
// //GEN-END:methods
/**
* Canonicalize the strings in this modlog.
* Used to reduce communication bandwidth when sending larger collections of modlogs via RMI.
*
* @param stringCanonicalizer the canonicalizer for the strings, null if none
* @param objectCanonicalizer the canonicalizer for the lazy object, null if none
*/
public void canonicalize(Canonicalizer stringCanonicalizer, Canonicalizer> objectCanonicalizer) {
if (stringCanonicalizer != null) {
txName = stringCanonicalizer.canonicalize(txName);
user = stringCanonicalizer.canonicalize(user);
message = stringCanonicalizer.canonicalize(message);
messageParameters = null; // force rebuild
}
if (objectCanonicalizer != null) {
lazyObject = objectCanonicalizer.canonicalize(lazyObject);
}
}
/**
* {@inheritDoc}.
* Overridden to set the db in lazyObject too (if unmarshalled from remote db)
*/
@Override
public void setSession(Session session) {
super.setSession(session);
if (lazyObject != null) {
lazyObject.setSession(session);
}
}
/**
* Clears the lazyObject.
* Necessary for replaying modlogs that should not copy the lazyObject
* to a remote db.
*/
public void clearLazyObject() {
lazyObject = null;
}
@Override
public ModificationLog readFromResultSetWrapper(ResultSetWrapper rs) {
ModificationLog log = super.readFromResultSetWrapper(rs);
if (log != null) {
// clear hidden attributes in case modlog is used more than once for resultSet...()
log.clearLazyObject();
log.messageParameters = null;
}
return log;
}
/**
* Check whether this modlog is modifying data.
*
* @return true if DELETE, DELETEALL, INSERT or UPDATE
*/
public boolean isModifyingData() {
return modType == DELETE ||
modType == INSERT ||
modType == UPDATE;
}
/**
* Returns whether this modlog belongs to a transaction.
*
* @return true if part of current transaction
*/
public boolean isLogOfTransaction() {
return getSession().isTxRunning();
}
/**
* Returns whether the modlog refers to the PDO at the destination side during replay.
*
* @return true if refer to destination db
*/
public boolean isDestinationReferringLog() {
return modType == DELETE;
}
/**
* {@inheritDoc}
*
* Overridden to check for deferred logging.
*/
@Override
public void saveObject() {
if (getSession().isRemote()) {
AbstractDbObject> oldLazyObject = lazyObject;
lazyObject = null; // don't transfer the lazyObject to the remote server
super.saveObject();
lazyObject = oldLazyObject; // restore lazyObject
}
else {
if (isLogOfTransaction()) {
if (getSession().isLogModificationDeferred()) {
newId(); // obtain an Id to trigger BEGIN only once
setSerial(getSerial() + 1); // increment serial as if it has been saved
}
else {
super.saveObject();
}
getSession().pushModificationLogOfTransaction(this);
}
else {
super.saveObject();
}
}
}
/**
* Gets the message parameters.
*
* @return the parameters, never null
* @throws ParseException if paramaters are malformed
*/
public ParameterString getMessageParameters() throws ParseException {
if (messageParameters == null) {
messageParameters = new ParameterString(getMessage());
}
return messageParameters;
}
/**
* Sets the message parameters.
*
* Updates the message field as well.
*
* @param messageParameters the message paramaters
*/
public void setMessageParameters(ParameterString messageParameters) {
this.messageParameters = messageParameters;
this.message = messageParameters == null ? null : messageParameters.toString();
}
/**
* Gets a message parameter.
*
* @param name the parameter's name
* @return the parameter's value
* @throws ParseException if parsing the message failed
*/
public String getMessageParameter(String name) throws ParseException {
return getMessageParameters().getParameter(name);
}
/**
* Sets a message parameter.
*
* Updates the message field as well.
*
* @param name the parameter's name
* @param value the parameter's value
* @throws ParseException if parsing the message failed
*/
public void setMessageParameter(String name, String value) throws ParseException {
getMessageParameters().setParameter(name, value);
this.message = messageParameters.toString();
}
@Override
public String toString() {
StringBuilder buf = new StringBuilder();
buf.append('<').append(getId()).append('/').append(modType).append(':');
if (user == null) {
buf.append("no-user");
}
else {
buf.append(user);
}
if (txId != 0) {
buf.append(',').append(txId);
if (txName != null) {
buf.append('/').append(txName);
}
}
buf.append(',');
if (when == null) {
buf.append("no-time");
}
else {
buf.append(when);
}
buf.append(">");
if (objectClassId != 0 && objectId != 0) {
buf.append(' ').append(PdoUtilities.getInstance().getPdoClassName(objectClassId))
.append('[').append(objectId).append('/').append(getSerial()).append(']');
}
if (message != null) {
buf.append(" \"").append(message).append("\"");
}
return buf.toString();
}
/**
* Gets the db object referenced by this ModificationLog.
* The object is lazily cached if the given db equals
* the db of this modlog.
*
* If the returned object is a {@link org.tentackle.pdo.PersistentObject},
* it's getPdo() (or pdo()) method will point to the PDO.
* Otherwise it's a low level {@link AbstractDbObject}.
*
* @param db is the db-connection from which to load the object.
* @return the object or null if not found.
*/
@SuppressWarnings("unchecked")
public AbstractDbObject> getDbObject(Db db) {
if (lazyObject != null && lazyObject.getSession().equals(db)) {
return lazyObject; // already lazily cached
}
String className = PdoUtilities.getInstance().getPdoClassName(objectClassId);
if (className == null) {
throw new PersistenceException(this, "unknown class id " + objectClassId);
}
try {
if (db.isRemote()) {
LOGGER.warning("inefficient remote object load while processing " + this);
}
lazyObject = null;
Class clazz = Class.forName(className);
if (PersistentDomainObject.class.isAssignableFrom(clazz)) {
PersistentDomainObject> lazyPdo = Pdo.create(clazz, db).select(objectId);
if (lazyPdo != null) {
lazyPdo.setDomainContext(lazyPdo.createValidContext());
lazyObject = (AbstractDbObject>) lazyPdo.getPersistenceDelegate();
}
}
else {
lazyObject = AbstractDbObject.newInstance(db, clazz).selectObject(objectId);
}
// load any lazy references that may be necessary for replay on the remote side
if (lazyObject != null) {
lazyObject.loadLazyReferences();
}
return lazyObject;
}
catch (Exception ex) {
throw new PersistenceException(this, "can't load object " + className + "[" + objectId + "]", ex);
}
}
/**
* Gets the object referenced by this ModificationLog.
* The object is lazily cached.
*
* If the returned object is a {@link org.tentackle.pdo.PersistentObject},
* it's getPdo() (or pdo()) method will point to the PDO.
* Otherwise it's a low level {@link AbstractDbObject}.
*
* @return the object or null if not found.
*/
public AbstractDbObject> getDbObject() {
return getDbObject(getSession());
}
/**
* Selects the next record to process.
* This is the first unprocessed modlog with the lowest ID.
*
* @return the modlog, null if no unprocessed log found
* @wurblet selectFirstUnprocessed DbSelectUnique --model=$mapfile processed:=:null +id
*/
// //GEN-BEGIN:selectFirstUnprocessed
public ModificationLog selectFirstUnprocessed() {
if (getSession().isRemote()) {
// invoke remote method
try {
ModificationLog obj = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectFirstUnprocessed();
getSession().applyTo(obj);
return obj;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_FIRST_UNPROCESSED_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 1, 0);
return sql.toString();
}
);
int ndx = getBackend().setLeadingSelectParameters(st, 1, 0);
st.setTimestamp(ndx++, null, true);
getBackend().setTrailingSelectParameters(st, ndx, 1, 0);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
return readFromResultSetWrapper(rs);
}
return null; // not found
}
}
private static final StatementId SELECT_FIRST_UNPROCESSED_STMT = new StatementId();
// //GEN-END:selectFirstUnprocessed
/**
* Selects the unprocessed modlogs as a result set.
*
* @return the resultset
* @wurblet resultSetUnprocessed DbSelectList --model=$mapfile --resultset processed:=:null +id
*/
// //GEN-BEGIN:resultSetUnprocessed
public ResultSetWrapper resultSetUnprocessed() {
PreparedStatementWrapper st = getPreparedStatement(RESULT_SET_UNPROCESSED_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, null, true);
return st.executeQuery();
}
private static final StatementId RESULT_SET_UNPROCESSED_STMT = new StatementId();
// //GEN-END:resultSetUnprocessed
/**
* Selects the next record to process greater than a given id.
*
* @param id the modlog id
* @return the modlog, null if no unprocessed log found
* @wurblet selectFirstUnprocessedGreater DbSelectUnique --model=$mapfile id:> processed:=:null +id
*/
// //GEN-BEGIN:selectFirstUnprocessedGreater
public ModificationLog selectFirstUnprocessedGreater(long id) {
if (getSession().isRemote()) {
// invoke remote method
try {
ModificationLog obj = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectFirstUnprocessedGreater(id);
getSession().applyTo(obj);
return obj;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_FIRST_UNPROCESSED_GREATER_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_ID);
sql.append(Backend.SQL_GREATER_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 1, 0);
return sql.toString();
}
);
int ndx = getBackend().setLeadingSelectParameters(st, 1, 0);
st.setLong(ndx++, id);
st.setTimestamp(ndx++, null, true);
getBackend().setTrailingSelectParameters(st, ndx, 1, 0);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
return readFromResultSetWrapper(rs);
}
return null; // not found
}
}
private static final StatementId SELECT_FIRST_UNPROCESSED_GREATER_STMT = new StatementId();
// //GEN-END:selectFirstUnprocessedGreater
/**
* Selects the first modlog since a given modification time.
*
* @param when the starting modification time
* @return the modlog if any exists
* @wurblet resultSetSince DbSelectList --model=$mapfile --resultset when:>= +id
*/
// //GEN-BEGIN:resultSetSince
public ResultSetWrapper resultSetSince(Timestamp when) {
PreparedStatementWrapper st = getPreparedStatement(RESULT_SET_SINCE_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_WHEN);
sql.append(Backend.SQL_GREATEROREQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, when);
return st.executeQuery();
}
private static final StatementId RESULT_SET_SINCE_STMT = new StatementId();
// //GEN-END:resultSetSince
/**
* Selects the first modlog with an ID greater than given ID.
*
* @param id the given ID
* @return the modlog if any exists
*
* @wurblet selectGreaterId DbSelectUnique --model=$mapfile id:> +id
*/
// //GEN-BEGIN:selectGreaterId
public ModificationLog selectGreaterId(long id) {
if (getSession().isRemote()) {
// invoke remote method
try {
ModificationLog obj = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectGreaterId(id);
getSession().applyTo(obj);
return obj;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_GREATER_ID_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_ID);
sql.append(Backend.SQL_GREATER_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 1, 0);
return sql.toString();
}
);
int ndx = getBackend().setLeadingSelectParameters(st, 1, 0);
st.setLong(ndx++, id);
getBackend().setTrailingSelectParameters(st, ndx, 1, 0);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
return readFromResultSetWrapper(rs);
}
return null; // not found
}
}
private static final StatementId SELECT_GREATER_ID_STMT = new StatementId();
// //GEN-END:selectGreaterId
/**
* Selects the first modlog with an ID greater than given ID.
*
* @param id the given ID
* @return the modlog if any exists
*
* @wurblet resultSetGreaterId DbSelectList --model=$mapfile --resultset id:> +id
*/
// //GEN-BEGIN:resultSetGreaterId
public ResultSetWrapper resultSetGreaterId(long id) {
PreparedStatementWrapper st = getPreparedStatement(RESULT_SET_GREATER_ID_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_ID);
sql.append(Backend.SQL_GREATER_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setLong(ndx++, id);
return st.executeQuery();
}
private static final StatementId RESULT_SET_GREATER_ID_STMT = new StatementId();
// //GEN-END:resultSetGreaterId
/**
* Selects the last processed modlog.
* This is the last processed modlog with the highest ID.
*
* @return the modlog, null if no processed log found
* @wurblet selectLastProcessed DbSelectUnique --model=$mapfile processed:!=:null -id
*/
// //GEN-BEGIN:selectLastProcessed
public ModificationLog selectLastProcessed() {
if (getSession().isRemote()) {
// invoke remote method
try {
ModificationLog obj = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectLastProcessed();
getSession().applyTo(obj);
return obj;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(this, e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_LAST_PROCESSED_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_NOTEQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTDESC);
getBackend().buildSelectSql(sql, false, 1, 0);
return sql.toString();
}
);
int ndx = getBackend().setLeadingSelectParameters(st, 1, 0);
st.setTimestamp(ndx++, null, true);
getBackend().setTrailingSelectParameters(st, ndx, 1, 0);
try (ResultSetWrapper rs = st.executeQuery()) {
if (rs.next()) {
return readFromResultSetWrapper(rs);
}
return null; // not found
}
}
private static final StatementId SELECT_LAST_PROCESSED_STMT = new StatementId();
// //GEN-END:selectLastProcessed
/**
* Gets the modlogs for a given object.
*
* @param objectClassId the object's class ID
* @param objectId the object's ID
* @return the list of modlogs
* @wurblet selectByObject DbSelectList --model=$mapfile processed:=:null objectClassId objectId +id
*/
// //GEN-BEGIN:selectByObject
public List selectByObject(int objectClassId, long objectId) {
if (getSession().isRemote()) {
// invoke remote method
try {
List list = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectByObject(objectClassId, objectId);
getSession().applyTo(list);
return list;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_BY_OBJECT_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTCLASSID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, null, true);
st.setInt(ndx++, objectClassId);
st.setLong(ndx++, objectId);
try (ResultSetWrapper rs = st.executeQuery()) {
List list = new ArrayList<>();
boolean derived = getClass() != ModificationLog.class;
while (rs.next()) {
ModificationLog obj = derived ? newInstance() : new ModificationLog(getSession());
obj = obj.readFromResultSetWrapper(rs);
if (obj != null) {
list.add(obj);
}
}
return list;
}
}
private static final StatementId SELECT_BY_OBJECT_STMT = new StatementId();
// //GEN-END:selectByObject
/**
* Selects all logs for a given user and type.
*
* @param user the username
* @param modType the modlog type
* @return the modlogs
* @wurblet selectByUserAndType DbSelectList --model=$mapfile processed:=:null user modType
*/
// //GEN-BEGIN:selectByUserAndType
public List selectByUserAndType(String user, char modType) {
if (getSession().isRemote()) {
// invoke remote method
try {
List list = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectByUserAndType(user, modType);
getSession().applyTo(list);
return list;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_BY_USER_AND_TYPE_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_USER);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_MODTYPE);
sql.append(Backend.SQL_EQUAL_PAR);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, null, true);
st.setString(ndx++, user);
st.setChar(ndx++, modType);
try (ResultSetWrapper rs = st.executeQuery()) {
List list = new ArrayList<>();
boolean derived = getClass() != ModificationLog.class;
while (rs.next()) {
ModificationLog obj = derived ? newInstance() : new ModificationLog(getSession());
obj = obj.readFromResultSetWrapper(rs);
if (obj != null) {
list.add(obj);
}
}
return list;
}
}
private static final StatementId SELECT_BY_USER_AND_TYPE_STMT = new StatementId();
// //GEN-END:selectByUserAndType
/**
* Checks if there are logs for a given user.
*
* @param user the user name
* @return true if there are logs
* @wurblet isReferencingUser DbIsReferencing --model=$mapfile processed:=:null user
*/
// //GEN-BEGIN:isReferencingUser
public boolean isReferencingUser(String user) {
if (getSession().isRemote()) {
// invoke remote method
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
isReferencingUser(user);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(IS_REFERENCING_USER_STMT,
() -> {
StringBuilder sql = createSelectIdInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_USER);
sql.append(Backend.SQL_EQUAL_PAR);
getBackend().buildSelectSql(sql, false, 1, 0);
return sql.toString();
}
);
int ndx = getBackend().setLeadingSelectParameters(st, 1, 0);
st.setTimestamp(ndx++, null, true);
st.setString(ndx++, user);
getBackend().setTrailingSelectParameters(st, ndx, 1, 0);
try (ResultSetWrapper rs = st.executeQuery()) {
return rs.next();
}
}
private static final StatementId IS_REFERENCING_USER_STMT = new StatementId();
// //GEN-END:isReferencingUser
/**
* Checks if there are logs for a given object.
*
* @param objectClassId the object's class ID
* @param objectId the object's ID
* @return true if there are logs
* @wurblet isReferencingObject DbIsReferencing --model=$mapfile objectClassId objectId
*/
// //GEN-BEGIN:isReferencingObject
public boolean isReferencingObject(int objectClassId, long objectId) {
if (getSession().isRemote()) {
// invoke remote method
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
isReferencingObject(objectClassId, objectId);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(IS_REFERENCING_OBJECT_STMT,
() -> {
StringBuilder sql = createSelectIdInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTCLASSID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTID);
sql.append(Backend.SQL_EQUAL_PAR);
getBackend().buildSelectSql(sql, false, 1, 0);
return sql.toString();
}
);
int ndx = getBackend().setLeadingSelectParameters(st, 1, 0);
st.setInt(ndx++, objectClassId);
st.setLong(ndx++, objectId);
getBackend().setTrailingSelectParameters(st, ndx, 1, 0);
try (ResultSetWrapper rs = st.executeQuery()) {
return rs.next();
}
}
private static final StatementId IS_REFERENCING_OBJECT_STMT = new StatementId();
// //GEN-END:isReferencingObject
/**
* Updates the processing timestamp by objectclass + objectid + serial less or equal than some value.
* Used to mark modlogs already processed for a given object. (poolkeeper)
*
* @param processed the new processing timestamp
* @param objectClassId the object class ID
* @param objectId the object id
* @param modType the modification type
* @param serial the object serial
* @return the number of modlogs updated
*
* @wurblet updateByObjectTypeSerial DbUpdateBy --model=$mapfile \
* processed:=:null objectClassId objectId modType serial:<= | processed
*/
// //GEN-BEGIN:updateByObjectTypeSerial
public int updateByObjectTypeSerial(Timestamp processed, int objectClassId, long objectId, char modType, long serial) {
if (getSession().isRemote()) {
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
updateByObjectTypeSerial(processed, objectClassId, objectId, modType, serial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(UPDATE_BY_OBJECT_TYPE_SERIAL_STMT,
() -> {
StringBuilder sql = createSqlUpdate();
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_WHEREALL);
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTCLASSID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_MODTYPE);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_SERIAL);
sql.append(Backend.SQL_LESSOREQUAL_PAR);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, processed, true);
st.setTimestamp(ndx++, null, true);
st.setInt(ndx++, objectClassId);
st.setLong(ndx++, objectId);
st.setChar(ndx++, modType);
st.setLong(ndx++, serial);
return st.executeUpdate();
}
private static final StatementId UPDATE_BY_OBJECT_TYPE_SERIAL_STMT = new StatementId();
// //GEN-END:updateByObjectTypeSerial
/**
* Deletes by objectclass + objectid + serial less or equal than some value.
* Used to remove modlogs already processed for a given object. (poolkeeper)
*
* @param objectClassId the object class ID
* @param objectId the object id
* @param modType the modification type
* @param serial the object serial
* @return the number of objects removed
*
* @wurblet deleteByObjectTypeSerial DbDeleteBy --model=$mapfile \
* processed:=:null objectClassId objectId modType serial:<=
*/
// //GEN-BEGIN:deleteByObjectTypeSerial
public int deleteByObjectTypeSerial(int objectClassId, long objectId, char modType, long serial) {
if (getSession().isRemote()) {
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
deleteByObjectTypeSerial(objectClassId, objectId, modType, serial);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(DELETE_BY_OBJECT_TYPE_SERIAL_STMT,
() -> {
StringBuilder sql = createDeleteAllSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTCLASSID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_OBJECTID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_MODTYPE);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_SERIAL);
sql.append(Backend.SQL_LESSOREQUAL_PAR);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, null, true);
st.setInt(ndx++, objectClassId);
st.setLong(ndx++, objectId);
st.setChar(ndx++, modType);
st.setLong(ndx++, serial);
return st.executeUpdate();
}
private static final StatementId DELETE_BY_OBJECT_TYPE_SERIAL_STMT = new StatementId();
// //GEN-END:deleteByObjectTypeSerial
/**
* Selects the transaction.
*
* @param txId the transaction id
* @return the modlogs of the transaction sorted by id
*
* @wurblet selectByTxId DbSelectList --model=$mapfile txId --bounded +id
*/
// //GEN-BEGIN:selectByTxId
public List extends ModificationLog> selectByTxId(long txId) {
if (getSession().isRemote()) {
// invoke remote method
try {
List extends ModificationLog> list = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectByTxId(txId);
getSession().applyTo(list);
return list;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_BY_TX_ID_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND);
sql.append(CN_TXID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setLong(ndx++, txId);
try (ResultSetWrapper rs = st.executeQuery()) {
List list = new ArrayList<>();
boolean derived = getClass() != ModificationLog.class;
while (rs.next()) {
ModificationLog obj = derived ? newInstance() : new ModificationLog(getSession());
obj = obj.readFromResultSetWrapper(rs);
if (obj != null) {
list.add(obj);
}
}
return list;
}
}
private static final StatementId SELECT_BY_TX_ID_STMT = new StatementId();
// //GEN-END:selectByTxId
/**
* Updates the processing timestamp of a transaction for all unprocessed modlogs.
*
* @param processed the new processing timestamp, null to set unprocessed
* @param txId the transaction id
* @return the number of updated modlogs
*
* @wurblet updateUnprocessedByTxId DbUpdateBy --model=$mapfile txId processed:=:null | processed
*/
// //GEN-BEGIN:updateUnprocessedByTxId
public int updateUnprocessedByTxId(Timestamp processed, long txId) {
if (getSession().isRemote()) {
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
updateUnprocessedByTxId(processed, txId);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(UPDATE_UNPROCESSED_BY_TX_ID_STMT,
() -> {
StringBuilder sql = createSqlUpdate();
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_WHEREALL);
sql.append(Backend.SQL_AND);
sql.append(CN_TXID);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, processed, true);
st.setLong(ndx++, txId);
st.setTimestamp(ndx++, null, true);
return st.executeUpdate();
}
private static final StatementId UPDATE_UNPROCESSED_BY_TX_ID_STMT = new StatementId();
// //GEN-END:updateUnprocessedByTxId
/**
* Deletes a transaction from the modlogs.
*
* @param txId the transaction id
* @return the number of deleted modlogs
*
* @wurblet deleteByTxId DbDeleteBy --model=$mapfile txId
*/
// //GEN-BEGIN:deleteByTxId
public int deleteByTxId(long txId) {
if (getSession().isRemote()) {
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
deleteByTxId(txId);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(DELETE_BY_TX_ID_STMT,
() -> {
StringBuilder sql = createDeleteAllSql();
sql.append(Backend.SQL_AND);
sql.append(CN_TXID);
sql.append(Backend.SQL_EQUAL_PAR);
return sql.toString();
}
);
int ndx = 1;
st.setLong(ndx++, txId);
return st.executeUpdate();
}
private static final StatementId DELETE_BY_TX_ID_STMT = new StatementId();
// //GEN-END:deleteByTxId
/**
* Deletes processed modlogs.
* Used to limit the backlog for recovery.
*
* Note: does not honour any transaction boundaries, so the first already processed
* transaction may be incomplete.
*
* @param processed the maximum processing date
* @return the number of deleted modlogs
* @wurblet deleteProcessed DbDeleteBy --model=$mapfile processed:!=:null processed:<=
*/
// //GEN-BEGIN:deleteProcessed
public int deleteProcessed(Timestamp processed) {
if (getSession().isRemote()) {
try {
return ((ModificationLogRemoteDelegate) getRemoteDelegate()).
deleteProcessed(processed);
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(DELETE_PROCESSED_STMT,
() -> {
StringBuilder sql = createDeleteAllSql();
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_NOTEQUAL_PAR);
sql.append(Backend.SQL_AND);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_LESSOREQUAL_PAR);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, null, true);
st.setTimestamp(ndx++, processed, true);
return st.executeUpdate();
}
private static final StatementId DELETE_PROCESSED_STMT = new StatementId();
// //GEN-END:deleteProcessed
/**
* Selects all modlogs that are unprocessed or modified up to a given timestamp.
* Used to replay modlogs after a crash.
*
* @param processed the minimum processing date, null = all
* @return the modlogs
* @wurblet selectUpTo DbSelectList --model=$mapfile processed:=:null or processed:>= +id
*/
// //GEN-BEGIN:selectUpTo
public List selectUpTo(Timestamp processed) {
if (getSession().isRemote()) {
// invoke remote method
try {
List list = ((ModificationLogRemoteDelegate) getRemoteDelegate()).
selectUpTo(processed);
getSession().applyTo(list);
return list;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(SELECT_UP_TO_STMT,
() -> {
StringBuilder sql = createSelectAllInnerSql();
sql.append(Backend.SQL_AND).append(Backend.SQL_LEFT_PARENTHESIS);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_EQUAL_PAR);
sql.append(Backend.SQL_OR);
sql.append(CN_PROCESSED);
sql.append(Backend.SQL_GREATEROREQUAL_PAR);
sql.append(Backend.SQL_RIGHT_PARENTHESIS);
sql.append(Backend.SQL_ORDERBY)
.append(CN_ID).append(Backend.SQL_SORTASC);
getBackend().buildSelectSql(sql, false, 0, 0);
return sql.toString();
}
);
int ndx = 1;
st.setTimestamp(ndx++, null, true);
st.setTimestamp(ndx++, processed, true);
try (ResultSetWrapper rs = st.executeQuery()) {
List list = new ArrayList<>();
boolean derived = getClass() != ModificationLog.class;
while (rs.next()) {
ModificationLog obj = derived ? newInstance() : new ModificationLog(getSession());
obj = obj.readFromResultSetWrapper(rs);
if (obj != null) {
list.add(obj);
}
}
return list;
}
}
private static final StatementId SELECT_UP_TO_STMT = new StatementId();
// //GEN-END:selectUpTo
/**
* Updates the processing timestamp.
*
* @param processed the processing timestamp, null to mark unprocessed
*/
public void updateProcessed(Timestamp processed) {
if (getSession().isRemote()) {
// invoke remote method
try {
((ModificationLogRemoteDelegate) getRemoteDelegate()).updateProcessed(processed, this);
this.processed = processed;
return;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(UPDATE_PROCESSES_STMTID,
() -> Backend.SQL_UPDATE + getTableName() + Backend.SQL_SET + CN_PROCESSED + Backend.SQL_EQUAL_PAR +
Backend.SQL_WHERE + CN_ID + Backend.SQL_EQUAL_PAR
);
st.setTimestamp(1, processed, true);
st.setLong(2, getId());
assertThisRowAffected(st.executeUpdate());
this.processed = processed;
}
private static final StatementId UPDATE_PROCESSES_STMTID = new StatementId();
/**
* Updates the processing timestamp and adds a comment to the message field.
*
* @param processed the processing timestamp, null to mark unprocessed
* @param comment the comment to add to the message, null if none
* @throws ParseException if message does not contain a valid parameter string
*/
public void updateDiagnostics(Timestamp processed, String comment) throws ParseException {
String parStr;
if (comment != null) {
ParameterString ps = new ParameterString(message);
ps.setParameter("comment", comment);
parStr = ps.toString();
}
else {
parStr = message; // unchanged
}
if (getSession().isRemote()) {
// invoke remote method
try {
((ModificationLogRemoteDelegate) getRemoteDelegate()).updateDiagnostics(processed, comment, this);
this.processed = processed;
this.message = parStr;
return;
}
catch (RemoteException e) {
throw PersistenceException.createFromRemoteException(getSession(), e);
}
}
// else: local mode
PreparedStatementWrapper st = getPreparedStatement(UPDATE_DIAGNOSTICS_STMTID,
() -> Backend.SQL_UPDATE + getTableName() + Backend.SQL_SET + CN_PROCESSED +
Backend.SQL_EQUAL_PAR_COMMA + CN_MESSAGE + Backend.SQL_EQUAL_PAR +
Backend.SQL_WHERE + CN_ID + Backend.SQL_EQUAL_PAR
);
int ndx = 1;
st.setTimestamp(ndx++, processed, true);
st.setString(ndx++, parStr);
st.setLong(ndx++, getId());
assertThisRowAffected(st.executeUpdate());
this.processed = processed;
this.message = parStr;
}
private static final StatementId UPDATE_DIAGNOSTICS_STMTID = new StatementId();
@Override
public DbObjectClassVariables getClassVariables() {
return CLASSVARIABLES;
}
/**
* Gets the message.
*
* @return optional informational or error message
*/
@Persistent("optional informational or error message")
public String getMessage() {
return message;
}
/**
* Sets the message.
*
* The lazy {@link #messageParameters} are cleared.
*
* @param message optional informational or error message
*/
@Persistent("optional informational or error message")
public void setMessage(String message) {
assertMutable();
this.message = message;
messageParameters = null; // force rebuild
}
/**
* Applies a modification to another db.
* The method is not static to allow overriding (e.g. to extend with more transaction types).
* Method is invoked within a tx, so no begin/commit/rollback necessary.
*
* The modlog to replay is passed as an argument because the replaying modlog is used
* to replay a list of modlogs and may perform some housekeeping and maintain state.
*
* @param modlog the modification log to replay
* @param toDb the db the logs will be applied to
*/
public void replay(ModificationLog modlog, Db toDb) {
LOGGER.fine("replaying modlog {0} to {1}", modlog, toDb);
// load the object (lazily during rmi, because the object is already switched to local db)
boolean retrieveFromDestination = modlog.isDestinationReferringLog();
Db db = retrieveFromDestination ? toDb : modlog.getSession();
AbstractDbObject> object = modlog.getDbObject(db);
if (object == null) {
handleMissingObject(modlog, toDb);
return; // if no exception
}
if (!retrieveFromDestination) {
// switch to other db
object.setSession(toDb);
}
try {
// perform any preprocessing
replayInitModification(modlog, object);
switch (modlog.getModType()) {
case INSERT:
replayInsert(modlog, object);
break;
case UPDATE:
replayUpdate(modlog, object);
break;
case DELETE:
replayDelete(modlog, object);
break;
default:
throw new PersistenceException(modlog, "illegal modType " + modlog.getModType() + " for replay");
}
// perform any post processing
replayFinishModification(modlog, object);
}
finally {
if (!retrieveFromDestination) {
object.setSession(modlog.getSession());
}
}
}
/**
* Perform preprocessing for replay.
*
* @param modlog the modification log
* @param object the object to insert
*/
public void replayInitModification(ModificationLog modlog, AbstractDbObject> object) {
try {
object.initModification(modlog.getModType());
}
catch (RuntimeException re) {
LOGGER.severe("replaying init-modification failed for " + modlog, re);
throw re;
}
}
/**
* Perform postprocessing for replay.
*
* @param modlog the modification log
* @param object the object to insert
*/
public void replayFinishModification(ModificationLog modlog, AbstractDbObject> object) {
try {
object.finishModification(modlog.getModType());
}
catch (RuntimeException re) {
LOGGER.severe("replaying finish modification failed for " + modlog, re);
throw re;
}
}
/**
* Replay an insert.
*
* @param modlog the modification log
* @param object the object to insert
*/
public void replayInsert(ModificationLog modlog, AbstractDbObject> object) {
try {
// set the modlog's serial (usually 1), in case multiple updates follow
object.setModificationLog(modlog);
long oldSerial = object.getSerial(); // keep old serial because of Canonilizer
object.setSerial(modlog.getSerial());
object.insertPlain();
object.setSerial(oldSerial);
}
catch (RuntimeException re) {
LOGGER.severe("replaying update failed for " + modlog, re);
throw re;
}
}
/**
* Replay an update.
*
* @param modlog the modification log
* @param object the object to update
*/
public void replayUpdate(ModificationLog modlog, AbstractDbObject> object) {
try {
object.setModificationLog(modlog);
if (object.getSerial() > modlog.getSerial()) {
// update only the serial
long oldSerial = object.getSerial(); // keep old serial because of Canonilizer
object.setSerial(modlog.getSerial() - 1);
object.updateSerial();
object.setSerial(oldSerial);
}
else if (object.getSerial() < modlog.getSerial()) {
throw new PersistenceException(modlog, "unexpected modlog serial " + modlog.getSerial() + " > object serial " + object.getSerial());
}
else {
// this is the final update
long oldSerial = object.getSerial();
object.setSerial(object.getSerial() - 1);
object.updatePlain();
object.setSerial(oldSerial);
}
}
catch (RuntimeException re) {
LOGGER.severe("replaying update failed for " + modlog, re);
throw re;
}
}
/**
* Replay a delete.
*
* @param modlog the modification log
* @param object the object to delete
*/
public void replayDelete(ModificationLog modlog, AbstractDbObject> object) {
try {
object.setModificationLog(modlog);
// set the modlog's serial, in case insert and further updates follow
long oldSerial = object.getSerial(); // keep old serial because of Canonilizer
object.setSerial(modlog.getSerial());
object.deletePlain();
object.setSerial(oldSerial);
}
catch (RuntimeException re) {
LOGGER.severe("replaying delete failed for " + modlog, re);
throw re;
}
}
/**
* Replay state shared between consecutive invocations of {@link #replay}.
*/
public static class ReplayState implements Serializable {
private static final long serialVersionUID = 1L;
/** saved logModificationAllowed. */
public boolean oldLogModificationAllowed;
/** transaction voucher. */
public long txVoucher;
/** the transaction id if BEGIN modlog found. */
public long pendingTxId;
/** the IDs of the replayed modlogs of this chunk. */
public long[] modlogIDs;
/** true if this is the first block. */
public boolean first;
/** true if this is the last block. */
public boolean last;
}
/**
* Replays a list of modlogs within a single transaction.
* It will also create new txId if the modlogs are copied.
* The method is not static to allow overriding (e.g. to extend with more transaction types).
*
* There may be several consecutive invocations within the same transaction.
* This allows some feedback during long running large modlog-lists.
*
* @param state the state of the preceeding invocation, never null
* @param modList the list of log objects from the source db
* @param copyLog true to copy the logs as well
* @param toDb the db the logs will be applied to
*
* @return the replay state, never null
*
* @throws PersistenceException if replay failed and transacation rolled back
*/
public ReplayState replay(ReplayState state, List extends ModificationLog> modList, boolean copyLog, Db toDb) {
if (state.first) {
state.oldLogModificationAllowed = toDb.isLogModificationAllowed();
toDb.setLogModificationAllowed(false); // don't log twice!
state.txVoucher = toDb.begin("replay");
}
try {
for (ModificationLog log: modList) {
// replay object's modification
if (log.getModType() != BEGIN && log.getModType() != COMMIT) { // ignore BEGIN/COMMIT
replay(log, toDb); // replay next modlog
}
if (copyLog) {
LOGGER.fine("copying modlog {0} to {1}", log, toDb);
log.clearLazyObject(); // no more needed, and don't change db in lazyObject and don't transfer to remote db
if (log.getModType() == BEGIN) {
state.pendingTxId = log.getId();
}
Session oldSession = log.getSession();
log.setSession(toDb);
log.setId(0);
log.newId(); // force to get a new Id for toDb!
if (state.pendingTxId != 0) {
log.setTxId(state.pendingTxId); // set the txId if a BEGIN was found
}
log.insertPlain();
if (log.getModType() == COMMIT) {
state.pendingTxId = 0; // clear pending id
}
log.setSession(oldSession);
}
}
if (state.last) {
toDb.commit(state.txVoucher);
toDb.setLogModificationAllowed(state.oldLogModificationAllowed);
}
state.modlogIDs = new long[modList.size()];
int ndx = 0;
for (ModificationLog log: modList) {
state.modlogIDs[ndx++] = log.getId();
}
return state;
}
catch (RuntimeException e) {
toDb.rollback(state.txVoucher);
toDb.setLogModificationAllowed(state.oldLogModificationAllowed);
if (e instanceof PersistenceException) {
throw e;
}
else {
throw new PersistenceException("replay modList failed", e);
}
}
}
/**
* Handles the case when an object to replay is not found.
*
* @param modlog the original modlog
* @param toDb the destination db
*/
protected void handleMissingObject(ModificationLog modlog, Db toDb) {
throw new PersistenceException(modlog.getModType() == DELETE ? modlog.getSession() : toDb,
"object " + PdoUtilities.getInstance().getPdoClassName(modlog.getObjectClassId()) +
"[" + modlog.getObjectId() + "] does not exist");
}
}