/* * JacORB - a free Java ORB * * Copyright (C) 1997-2014 Gerald Brose / The JacORB Team. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the Free * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ package org.jacorb.orb.giop; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import org.jacorb.config.Configuration; import org.jacorb.orb.CodeSet; import org.jacorb.orb.ParsedIOR; import org.omg.CONV_FRAME.CodeSetComponentInfo; import org.slf4j.Logger; /** * @author Nicolas Noffke */ public class ClientConnection implements ReplyListener, ConnectionListener { private final GIOPConnection connection; private final org.omg.CORBA.ORB orb; private final Map<Integer, ReplyPlaceholder> replies; /** * <code>sasContexts</code> is used to support for SAS Stateful contexts. */ private final Map<String, Long> sasContexts; private static long last_client_context_id = 0; /** * <code>client_count</code> denotes how many clients use this connection. */ private int client_count = 0; /** * <code>id_count</code> is used to generate request ids. */ private int id_count = 0; private final ClientConnectionManager conn_mg; private final boolean client_initiated; private boolean listen_point_list_sent = false; /** * <code>gracefulStreamClose</code> indicates if the stream has been closed * gracefully, i.e. by a CloseConnection message. This will trigger a * remarshaling of all pending messages. */ private boolean gracefulStreamClose; /** * <code>registeredProfile</code> indicates the profile that was used for * registering with the ClientConnectionManager. In case of BiDirIIOP it is * NOT equal to the transports profile. */ private final org.omg.ETF.Profile registeredProfile ; /** * <code>logger</code> is the logger for this object. */ private final Logger logger; /** * <code>ignoreComponentInfo</code> defaults to off. If jacorb.codeset * is turned on then it will NOT ignore component profiles. If * jacorb.codeset is turned off then this allows it to ignore codeset profiles * in an IOR. */ private final boolean ignoreComponentInfo; public ClientConnection( GIOPConnection connection, org.omg.CORBA.ORB orb, ClientConnectionManager conn_mg, org.omg.ETF.Profile registeredProfile, boolean client_initiated ) { this.connection = connection; this.orb = orb; this.conn_mg = conn_mg; this.registeredProfile = registeredProfile; this.client_initiated = client_initiated; final Configuration configuration = ((org.jacorb.orb.ORB)orb).getConfiguration(); logger = configuration.getLogger("org.jacorb.giop.conn"); ignoreComponentInfo = ! (configuration.getAttributeAsBoolean("jacorb.codeset", true)); //For BiDirGIOP, the connection initiator may only generate //even valued request ids, and the other side odd valued //request ids. Therefore, we always step the counter by 2, so //we always get only odd or even ids depending on the counters //initial value. if( ! client_initiated ) { id_count = 1; } connection.setReplyListener( this ); connection.setConnectionListener( this ); replies = new HashMap<Integer, ReplyPlaceholder>(); sasContexts = new HashMap<String, Long>(); } public final GIOPConnection getGIOPConnection() { return connection; } /** * Get the profile that was used for registering with the * ClientConnectionManager. In case of BiDirIIOP it is NOT equal * to the transports profile. */ public org.omg.ETF.Profile getRegisteredProfile() { return registeredProfile; } public void setCodeSet( ParsedIOR pior ) { if( isTCSNegotiated() ) { //if already negotiated, do nothing return; } //if the other side only talks GIOP 1.0, don't send a codeset //context and don't try again if( pior.getEffectiveProfile().version().minor == 0 ) { connection.markTCSNegotiated(); return; } CodeSetComponentInfo info = pior.getCodeSetComponentInfo(); if (info != null && !ignoreComponentInfo) { connection.markTCSNegotiated(); // even if this aborts, we should not try negotiating again. CodeSet c1 = CodeSet.getNegotiatedCodeSet((org.jacorb.orb.ORB)orb, info, /* wide */ false ); CodeSet c2 = CodeSet.getNegotiatedCodeSet((org.jacorb.orb.ORB)orb, info, /* wide */ true ); connection.setCodeSets( c1, c2); logger.info ("Negotiated char codeset of " + c1 + " and wchar of " + c2); } else { logger.debug("No CodeSetComponentInfo in IOR. Will use default CodeSets" ); //If we can't find matching codesets, we still mark the //GIOPConnection as negotiated, so the following requests //will not always try to select a codeset again. /* ****** until the ETF spec is ammended to include components within the base Profile type, then this is going to be problem. So rather than not setting the codeset component, we should pick reasonable default values and send those. */ connection.markTCSNegotiated(); } } public boolean isTCSNegotiated() { return connection.isTCSNegotiated(); } public CodeSet getTCS() { return connection.getTCS(); } public CodeSet getTCSW() { return connection.getTCSW(); } public synchronized int getId() { int result = id_count; //if odd or even is determined by the starting value of //id_count id_count += 2; return result; } /** * Increments the number of clients. */ public synchronized void incClients() { client_count++; } /** * This method decrements the number of clients. * * @return a <code>boolean</code> value, true if client_count is zero. */ public synchronized boolean decClients() { boolean result = false; client_count--; if (client_count == 0 ) { result = true; } return result; } /** * Returns the number of clients currently using this connection. */ public int numClients() { return client_count; } public boolean isClientInitiated() { return client_initiated; } /** * Used as part of the configuration of a bidirectional connection * Not only returns the value of the listen point list sent flag, * it also sets the value. This is effectively a one-shot method * that ensures only the first caller gets the opportunity to send * a listen point list as part of a BiDirIIOP context. */ public synchronized boolean isListenPointListSent() { boolean rtn = listen_point_list_sent; listen_point_list_sent = true; return rtn; } /** * The request_id parameter is only used, if response_expected. */ public void sendRequest( MessageOutputStream outputStream, ReplyPlaceholder placeholder, int request_id, boolean response_expected ) { Integer key = Integer.valueOf(request_id); synchronized( replies ) { replies.put( key, placeholder ); } try { sendRequest( outputStream, response_expected ); } catch( org.omg.CORBA.SystemException e ) { //remove reply receiver from list //because there will be no response to this request synchronized( replies ) { replies.remove( key ); } throw e; } } public void sendRequest( MessageOutputStream outputStream, boolean response_expected ) { try { connection.sendRequest( outputStream, response_expected ); } catch (java.io.IOException e) { if (logger.isDebugEnabled()) { logger.debug("IOException", e); } throw new org.omg.CORBA.COMM_FAILURE (0, org.omg.CORBA.CompletionStatus.COMPLETED_MAYBE); } } /** * called from Delegate/ConnectionManagement etc. */ public void close() { connection.close(); } /** * Return the state of the underlying GIOP connection. */ public boolean isClosed() { return connection.do_close; } /** * Operations from ReplyListener */ @Override public void replyReceived( byte[] reply, GIOPConnection connection ) { connection.decPendingMessages(); Integer key = Integer.valueOf(Messages.getRequestId( reply )); ReplyPlaceholder placeholder = null; synchronized( replies ) { placeholder = replies.remove( key ); } if( placeholder != null ) { ReplyInputStream ris = new ReplyInputStream (orb, reply); ris.setCodeSet (this.getTCS(),this.getTCSW()); //this will unblock the waiting thread placeholder.replyReceived(ris); } else { if (logger.isWarnEnabled()) { logger.warn("Received reply for unknown request id: " + key); } } } @Override public void locateReplyReceived( byte[] reply, GIOPConnection connection ) { connection.decPendingMessages(); Integer key = Integer.valueOf(Messages.getRequestId( reply )); ReplyPlaceholder placeholder = null; synchronized( replies ) { placeholder = replies.remove( key ); } if( placeholder != null ) { //this will unblock the waiting thread placeholder.replyReceived( new LocateReplyInputStream( orb, reply )); } else { if (logger.isWarnEnabled()) { logger.warn("Received reply for unknown request id: " + key); } } } /** * Received a CloseConnection message. Remarshal all pending * messages. The close mechanism will be invoked separately by the * actual closing of the Transport and will trigger the * remarshaling. */ @Override public void closeConnectionReceived( byte[] close_conn, GIOPConnection connection ) { if (logger.isInfoEnabled()) { logger.info("Received CloseConnection on " + connection.toString()); } if( client_initiated ) { gracefulStreamClose = true; ((ClientGIOPConnection) connection).closeAllowReopen(); //since this is run on the message receptor thread itself, it //will not try to read again after returning, because it just //closed the transport itself. Therefore, no exception goes //back up into the GIOPConnection, where streamClosed() will //be called. Ergo, we need to call streamClosed() ourselves. streamClosed(); } } /** * Operations from ConnectionListener * used for upcalls from GIOPConnection */ @Override public void connectionClosed() { if( ! client_initiated ) { //if this is a server side BiDir connection, it will stay //pooled in the ClientConnectionManager even if no Delegate is //associated with it. Therefore, it has to be removed when //the underlying connection closed. conn_mg.removeConnection( this ); } streamClosed(); } /** * the transport has been * removed underneath the GIOP layer */ @Override public void streamClosed() { synchronized( replies ) { if( replies.size() > 0 ) { if( gracefulStreamClose ) { if (logger.isDebugEnabled()) { logger.debug("Stream closed. Will remarshal " + replies.size() + " messages" ); } } else { if (logger.isWarnEnabled()) { logger.warn("Abnormal connection termination. Lost " + replies.size() + " outstanding replie(s)!"); } } Iterator<ReplyPlaceholder> entries = replies.values().iterator(); ReplyPlaceholder placeholder; while( entries.hasNext() ) { placeholder = entries.next(); if( gracefulStreamClose ) { placeholder.retry(); } else { placeholder.cancel(); } entries.remove(); } } } gracefulStreamClose = false; } public org.omg.ETF.Profile get_server_profile() { return connection.getTransport().get_server_profile(); } public long cacheSASContext(byte[] client_authentication_token) { long client_context_id = 0; String key = new String(client_authentication_token); synchronized ( sasContexts ) { if (!sasContexts.containsKey(key)) { // new context client_context_id = ++last_client_context_id; sasContexts.put(key, Long.valueOf(client_context_id)); client_context_id = -client_context_id; } else { // reuse cached context client_context_id = sasContexts.get(key).longValue(); } } return client_context_id; } public long purgeSASContext(long client_context_id) { synchronized ( sasContexts ) { Iterator<String> entries = sasContexts.keySet().iterator(); while( entries.hasNext() ) { Object key = entries.next(); if (sasContexts.get(key).longValue() != client_context_id) { continue; } entries.remove(); break; } } return client_context_id; } }// ClientConnection