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

com.threerings.presents.server.ClientResolver Maven / Gradle / Ivy

//
// $Id$
//
// Narya library - tools for developing networked games
// Copyright (C) 2002-2012 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.server;

import java.util.List;

import com.google.common.collect.Lists;
import com.google.inject.Inject;

import com.samskivert.util.Invoker;

import com.threerings.util.Name;

import com.threerings.presents.annotation.MainInvoker;
import com.threerings.presents.data.ClientObject;
import com.threerings.presents.dobj.RootDObjectManager;

import static com.threerings.presents.Log.log;

/**
 * Used to resolve client data when a user starts a session (or when some other entity needs access
 * to a client object). Implementations will want to extend this class and override {@link
 * #resolveClientData}, making the necessary database calls and populating the client object
 * appropriately.
 */
public class ClientResolver extends Invoker.Unit
{
    /**
     * Thrown during resolution if the client disconnects.
     */
    public static class ClientDisconnectedException extends Exception
    {
    }

    /**
     * Initializes this instance.
     *
     * @param username the username of the user to be resolved.
     */
    public void init (Name username)
    {
        _username = username;
    }

    /**
     * Adds a resolution listener to this active resolver.
     */
    public void addResolutionListener (ClientResolutionListener listener)
    {
        _listeners.add(listener);
    }

    /**
     * Creates the {@link ClientObject} derived class that should be created to kick off the
     * resolution process.
     */
    public ClientObject createClientObject ()
    {
        return new ClientObject();
    }

    /**
     * Creates a record that will be maintained only on the server to track client related bits.
     */
    public ClientLocal createLocalAttribute ()
    {
        return new ClientLocal();
    }

    /**
     * Called once our client object is registered with the distributed object system.
     */
    public void objectAvailable (ClientObject object)
    {
        // we've got our object, so shunt ourselves over to the invoker thread to perform database
        // loading
        _clobj = object;
        _invoker.postUnit(this);
    }

    @Override
    public boolean invoke ()
    {
        try {
            // allow our derived class to do its database loads
            resolveClientData(_clobj);
        } catch (Exception cause) {
            // keep this around until we're back on the dobj thread
            _failure = cause;
        }
        return true;
    }

    @Override
    public void handleResult ()
    {
        // if we haven't failed, finish resolution on the dobj thread
        if (_failure == null) {
            try {
                finishResolution(_clobj);
            } catch (Exception e) {
                _failure = e;
            }
        }

        // if we still haven't failed, then we're good to go
        if (_failure == null) {
            // and let the listeners in on the secret as well
            reportSuccess();

        } else {
            // destroy the dangling user object
            _omgr.destroyObject(_clobj.getOid());

            // let our listener know that we're hosed
            reportFailure(_failure);
        }
    }

    @Override
    public String toString ()
    {
        return "ClientResolver:" + _username;
    }

    /**
     * This method is called on the invoker thread which means that it can do things like blocking
     * database requests and generally whatever is necessary to load up all the client data that is
     * desired by the implentation system. Any exceptions that are thrown will be caught and
     * reported as a failure to the client resolution listener.
     */
    protected void resolveClientData (ClientObject clobj)
        throws Exception
    {
        // fill in the username
        clobj.username = _username;
    }

    /**
     * This method is called on the dobj thread after resolveClientData returns normally, it should
     * finish populating the client object with any data that is NOT loaded from a database.
     */
    protected void finishResolution (ClientObject clobj)
    {
        // nothing to do by default
    }

    /**
     * Reports success to our resolution listeners.
     */
    protected void reportSuccess ()
    {
        for (int ii = 0, ll = _listeners.size(); ii < ll; ii++) {
            ClientResolutionListener crl = _listeners.get(ii);
            try {
                // add a reference for each listener
                _clobj.reference();
                crl.clientResolved(_username, _clobj);
            } catch (Exception e) {
                log.warning("Client resolution listener choked in clientResolved() " + crl, e);
            }
        }
    }

    /**
     * Reports failure to our resolution listeners.
     */
    protected void reportFailure (Exception cause)
    {
        for (int ii = 0, ll = _listeners.size(); ii < ll; ii++) {
            ClientResolutionListener crl = _listeners.get(ii);
            try {
                crl.resolutionFailed(_username, cause);
            } catch (Exception e) {
                log.warning("Client resolution listener choked in resolutionFailed()", "crl", crl,
                            "username", _username, "cause", cause, e);
            }
        }
    }

    /**
     * Throws an exception if the client being resolved is no longer connected.
     */
    protected void enforceConnected ()
        throws ClientDisconnectedException
    {
        if (_clmgr.getClient(_username) == null) {
            throw new ClientDisconnectedException();
        }
    }

    /** The name of the user whose client object is being resolved. */
    protected Name _username;

    /** The entities to notify of success or failure. */
    protected List _listeners = Lists.newArrayList();

    /** The resolving client object. */
    protected ClientObject _clobj;

    /** A place to keep an exception around for a moment. */
    protected Exception _failure;

    // dependencies
    protected @Inject @MainInvoker Invoker _invoker;
    protected @Inject RootDObjectManager _omgr;
    protected @Inject ClientManager _clmgr;
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy