
com.threerings.presents.util.PersistingUnit Maven / Gradle / Ivy
//
// $Id: PersistingUnit.java 6533 2011-03-15 18:17:09Z mdb $
//
// Narya library - tools for developing networked games
// Copyright (C) 2002-2011 Three Rings Design, Inc., All Rights Reserved
// http://code.google.com/p/narya/
//
// 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 com.threerings.presents.util;
import com.samskivert.util.ArrayUtil;
import com.samskivert.util.Invoker;
import com.threerings.presents.client.InvocationService;
import com.threerings.presents.data.InvocationCodes;
import com.threerings.presents.server.InvocationException;
import static com.threerings.presents.Log.log;
/**
* Simplifies a common pattern which is to post an {@link Invoker} unit which does some database
* operation and then calls back to an {@code InvocationService.InvocationListener} of some
* sort. If the database operation fails, the error will be logged and the result listener will be
* replied to with {@link InvocationCodes#INTERNAL_ERROR}.
*/
public abstract class PersistingUnit extends Invoker.Unit
{
public PersistingUnit (InvocationService.InvocationListener listener)
{
this("UnknownPersistingUnit", listener);
}
public PersistingUnit (String name, InvocationService.InvocationListener listener)
{
super(name);
_listener = listener;
}
/**
* Creates a persisting unit with the supplied name and listener and a set of key/value pairs
* that will be included in the failure message if this unit fails.
*/
public PersistingUnit (String name, InvocationService.InvocationListener listener,
Object... args)
{
super(name);
if (args != null && args.length % 2 == 1) {
throw new IllegalArgumentException("Missing key or value in key/value pair arguments.");
}
_listener = listener;
_args = args;
}
/**
* This method is where the unit performs its persistent actions. Any persistence exception
* will be caught and logged along with the output from {@link #getFailureMessage}, if any.
*/
public abstract void invokePersistent ()
throws Exception;
/**
* Handles the success case, which by default posts a response to a ConfirmListener.
* If you need something else, or to repond to a ResultListener, you'll need to override this.
*/
public void handleSuccess ()
{
if (_listener instanceof InvocationService.ConfirmListener) {
reportRequestProcessed();
} else if (_listener instanceof InvocationService.ResultListener && _resultSet) {
reportRequestProcessed(_result);
}
}
/**
* Handles the failure case by logging the error and reporting an internal error to the
* listener.
*/
public void handleFailure (Exception error)
{
if (error instanceof InvocationException) {
_listener.requestFailed(error.getMessage());
} else {
if (_args != null) {
log.warning(getFailureMessage(), ArrayUtil.append(_args, error));
} else {
log.warning(getFailureMessage(), error);
}
_listener.requestFailed(InvocationCodes.INTERNAL_ERROR);
}
}
@Override // from Invoker.Unit
public boolean invoke ()
{
try {
invokePersistent();
} catch (Exception pe) {
_error = pe;
}
return true;
}
@Override // from Invoker.Unit
public void handleResult ()
{
if (_error != null) {
handleFailure(_error);
} else {
handleSuccess();
}
}
/**
* If the listener is known to be a ConfirmListener, this will cast it and report that the
* request was processed.
*/
protected void reportRequestProcessed ()
{
((InvocationService.ConfirmListener)_listener).requestProcessed();
}
/**
* If the listener is known to be a ResultListener, this will cast it and report that the
* request was processed.
*/
protected void reportRequestProcessed (Object result)
{
((InvocationService.ResultListener)_listener).requestProcessed(result);
}
/**
* Provides a custom failure message in the event that the persistent action fails. This will
* be logged along with the exception.
*/
protected String getFailureMessage ()
{
return this + " failed.";
}
/**
* If the service listener is a ResultListener and this method is called, then the given
* result will be passed along to the listener on success.
*/
protected void setResult (Object result)
{
_resultSet = true;
_result = result;
}
protected InvocationService.InvocationListener _listener;
protected Exception _error;
protected Object[] _args;
protected boolean _resultSet;
protected Object _result;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy