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

org.tentackle.persist.rmi.RemoteDbConnectionImpl 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.rmi;

import java.io.Serializable;
import java.net.BindException;
import java.rmi.Remote;
import java.rmi.RemoteException;
import java.rmi.server.RMIClientSocketFactory;
import java.rmi.server.RMIServerSocketFactory;
import java.rmi.server.UnicastRemoteObject;
import org.tentackle.common.ExceptionHelper;
import org.tentackle.io.ServerSocketConfigurator;
import org.tentackle.io.ServerSocketConfiguratorHolder;
import org.tentackle.log.Logger;
import org.tentackle.log.LoggerFactory;
import org.tentackle.pdo.Session;
import org.tentackle.pdo.SessionInfo;
import org.tentackle.pdo.VersionInfoIncompatibleException;



/**
 * Server side RMI-connection implementation.
 * 

* Overview of what happens in a Tentackle RMI application server: *

    *
  • client gets a reference to a RemoteDbConnection in the server
  • *
  • client invokes login() on the connection and gets a reference to * a RemoteDbSession.
  • *
  • each session runs in its own thread (server & client) and the server * session gets its own Db.
  • *
  • all further client requests work through the session object and * the remote delegates created within the session
  • *
  • the client invokes logout() to close the session and Db.
  • *
  • in case the client terminates abnormally (without invoking logout()) * a timeout runs logout() via finalize().
  • *
* @author harald */ abstract public class RemoteDbConnectionImpl extends RemoteServerObject implements RemoteDbConnection { private static final long serialVersionUID = -1008588987968414154L; /** * logger for this class. */ private static final Logger LOGGER = LoggerFactory.getLogger(RemoteDbConnectionImpl.class); private final DbServer server; // the server /** * Creates a connection.
* * @param server the DbServer * @param port the tcp-port, 0 = system default * @param csf the client socket factory, null = system default * @param ssf the server socket factory, null = system default * @throws RemoteException * @see DbServer */ public RemoteDbConnectionImpl(DbServer server, int port, RMIClientSocketFactory csf, RMIServerSocketFactory ssf) throws RemoteException { super(port, csf, ssf); this.server = server; } /** * Gets the DbServer. * * @return the db server */ public DbServer getServer() { return server; } /** * Gets the session timeout count. * * @param session the session attached to the {@link RemoteDbSession} * @param clientInfo the UserInfo from the client * @param serverInfo the UserInfo to establish the connection to the database server * @return the timeout */ public int getSessionTimeout(Session session, SessionInfo clientInfo, SessionInfo serverInfo) { return getServer().getSessionTimeout(); } /** * Gets the tcp port for this connection. * * @param session the session attached to the {@link RemoteDbSession} * @param clientInfo the UserInfo from the client * @param serverInfo the UserInfo to establish the connection to the database server * @return the tcp-port for this connection, 0 = system default */ public int getPort(Session session, SessionInfo clientInfo, SessionInfo serverInfo) { return getRMIPort(); } /** * Gets the client socket factory according to the userinfos. * * @param session the session attached to the {@link RemoteDbSession} * @param clientInfo the UserInfo from the client * @param serverInfo the UserInfo to establish the connection to the database server * @return the client socket factory for this connection, null = system default */ public RMIClientSocketFactory getClientSocketFactory(Session session, SessionInfo clientInfo, SessionInfo serverInfo) { return getRMICsf(); } /** * Gets the server socket factory. * * @param session the session attached to the {@link RemoteDbSession} * @param clientInfo the UserInfo from the client * @param serverInfo the UserInfo to establish the connection to the database server * @return the server socket factory for this connection, null = system default */ public RMIServerSocketFactory getServerSocketFactory(Session session, SessionInfo clientInfo, SessionInfo serverInfo) { return getRMISsf(); } /** * Checks the client's version information. *

* The default implementation does nothing. * It is invoked from {@link org.tentackle.persist.Db#open()} for remote connections. * * @param clientVersionInfo the server's version info from {@link org.tentackle.persist.rmi.RemoteDbConnection}. * @throws VersionInfoIncompatibleException if versions are not compatible */ public void checkClientVersionInfo(Serializable clientVersionInfo) throws VersionInfoIncompatibleException { // default is ok } /** * Creates the session.
* Needs to be implemented by the application. * * @param clientInfo the client info (login info) * @return the created session * @throws RemoteException if creation failed */ abstract public RemoteDbSession createSession(SessionInfo clientInfo) throws RemoteException; /** * Exports the given remote object. *

* Notice that the delegate must not extend {@link UnicastRemoteObject}! * * @param remoteObject the object to export * @param port the port to export the object on, 0 if auto * @param csf the client-side socket factory for making calls to the remote object, null if system default * @param ssf the server-side socket factory for receiving remote calls, null if system default * @return the effective port, 0 if system default * @throws RemoteException if export failed */ public int exportRemoteObject(Remote remoteObject, int port, RMIClientSocketFactory csf, RMIServerSocketFactory ssf) throws RemoteException { if (remoteObject instanceof UnicastRemoteObject) { throw new RemoteException("delegate " + remoteObject.getClass().getName() + " must not extend UnicastRemoteObject!"); } int portRange = 0; if (ssf instanceof ServerSocketConfiguratorHolder) { ServerSocketConfiguratorHolder sscd = (ServerSocketConfiguratorHolder) ssf; ServerSocketConfigurator ssc = sscd.getSocketConfigurator(); if (ssc != null) { portRange = ssc.getPortRange(); if (ssc.getPort() > 0) { port = ssc.getPort(); } } } if (portRange < 1 || port == 0) { portRange = 1; } RemoteException lastEx = null; int effectivePort = -1; int maxPort = port + portRange; for (int p = port; p < maxPort; p++) { try { UnicastRemoteObject.exportObject(remoteObject, p, csf, ssf); effectivePort = p; break; } catch (RemoteException ex) { if (ExceptionHelper.extractException(BindException.class, true, ex) == null) { throw ex; // some other exception } // BindException: try next port lastEx = ex; } } if (effectivePort == -1) { // export failed for all ports in range throw lastEx; // cannot be null } LOGGER.fine("{0} exported on port {1}", remoteObject, effectivePort); return effectivePort; } /** * Exports the given remote object. * * @param remoteObject the object to unexport */ public void unexportRemoteObject(Remote remoteObject) throws RemoteException { UnicastRemoteObject.unexportObject(remoteObject, true); LOGGER.fine("{0} unexported", remoteObject); } /** * Overridden to detect unwanted garbage collection as this * should never happen. * If using DbServer, this will not happen because it keeps a reference * to the connection object. */ @Override protected void finalize() throws Throwable { try { LOGGER.warning("Connection object " + getClass().getName() + " finalized unexpectedly:\n" + this); } catch (Exception ex) { // don't stop finalization if just the logging failed } finally { super.finalize(); } } // ------------------ implements RemoteDbConnection ----------------- @Override public Serializable getServerVersionInfo() throws RemoteException { return null; } @Override public RemoteDbSession login(SessionInfo clientInfo) throws RemoteException { // check client compatibility checkClientVersionInfo(clientInfo.getClientVersionInfo()); // create the session RemoteDbSession session = createSession(clientInfo); // export the session if (session instanceof Exportable) { ((Exportable) session).exportMe(); } return session; } @Override public void logout(RemoteDbSession session) throws RemoteException { session.close(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy