/* * ALMA - Atacama Large Millimiter Array * (c) European Southern Observatory, 2005 * Copyright by ESO (in the framework of the ALMA collaboration), * All rights reserved * * 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 alma.acs.component.client; import java.util.HashMap; import java.util.Map; import java.util.concurrent.ThreadFactory; import java.util.logging.Logger; import com.cosylab.CDB.DAL; import com.cosylab.CDB.DALHelper; import si.ijs.maci.Client; import alma.JavaContainerError.wrappers.AcsJContainerServicesEx; import alma.acs.container.AcsManagerProxy; import alma.acs.container.CleaningDaemonThreadFactory; import alma.acs.container.ContainerServices; import alma.acs.container.ContainerServicesImpl; import alma.acs.container.corba.AcsCorba; import alma.acs.logging.AcsLogger; /** * Special version of <code>ComponentClient</code>, which gives more power to specialized applications such as the OMC Gui ("Exec"). * Using some of these methods can be dangerous to the stability of the entire system. * <p> * If you find that your application needs to use <code>AdvancedComponentClient</code> instead of <code>ComponentClient</code>, * better check with the ACS team if there really is no other way. * * @author hsommer */ public class AdvancedComponentClient extends ComponentClient { private Map<ContainerServicesImpl, AcsManagerProxy> additionalContainerServices = new HashMap<ContainerServicesImpl, AcsManagerProxy>(); public AdvancedComponentClient(Logger logger, String managerLoc, String clientName) throws Exception { super(logger, managerLoc, clientName); } /** * This ctor's implementation comes already from the base class for practical reasons, but there it is not public. * Only here it becomes available to client applications. */ public AdvancedComponentClient(Logger logger, String managerLoc, String clientName, AcsCorba externalAcsCorba) throws Exception { super(logger, managerLoc, clientName, externalAcsCorba); } /** * Factory method for additional container service instances. This method should only be used by specialized clients * such as the OMC GUI which needs independent ContainerServices instances for the plug-ins it runs. * <p> * Make sure to call {@link #destroyContainerServices(ContainerServices)} when done with the new CS. * * @param clientName * name for {@link ContainerServices#getName()} * @param csLogger * logger to be used internally by the new ContainerServices instance (which is different from the Logger * returned in {@link ContainerServices#getLogger()}). * Since ACS 8.0 it is recommended to supply an {@link AcsLogger} instead of a plain JDK Logger because a * plain Logger will have to be wrapped inside this method. */ public ContainerServices createContainerServices(String clientName, Logger csLogger) throws AcsJContainerServicesEx { if (clientName == null) { throw new IllegalArgumentException("clientName must not be null"); } if (csLogger == null) { throw new IllegalArgumentException("csLogger must not be null"); } try { // wrap csLogger if necessary AcsLogger acsLogger = AcsLogger.fromJdkLogger(csLogger, null); ThreadFactory threadFactory = new CleaningDaemonThreadFactory(clientName, csLogger); // separately log in to the manager to get a new client handle. // TODO: if this does not work, then we need a way to get a new handle from manager without logging in separately. // Note that when activating components, the container receives the new handle directly from the manager. AcsManagerProxy acsManagerProxy = m_acsManagerProxy.createInstance(); ManagerClient clImpl = new ManagerClient(clientName, acsLogger); Client managerClient = clImpl._this(acsCorba.getORB()); acsManagerProxy.loginToManager(managerClient, 0); int clientHandle = acsManagerProxy.getManagerHandle(); DAL cdb = DALHelper.narrow(m_acsManagerProxy.get_service("CDB", false)); ContainerServicesImpl cs = new ContainerServicesImpl(acsManagerProxy, cdb, acsCorba.getRootPOA(), acsCorba, acsLogger, clientHandle, clientName, null, threadFactory); additionalContainerServices.put(cs, acsManagerProxy); return cs; } catch (Throwable thr) { throw new AcsJContainerServicesEx(thr); } } /** * "un-factory" method which inverts {@link #createContainerServices(String, Logger)}. * @param cs ContainerServices instance created by {@link #createContainerServices(String, Logger)}. */ public void destroyContainerServices(ContainerServices cs) throws AcsJContainerServicesEx { if (!additionalContainerServices.containsKey(cs)) { AcsJContainerServicesEx ex = new AcsJContainerServicesEx(); ex.setContextInfo("The given ContainerServices object was not created by this AdvancedComponentClient!"); throw ex; } try { ContainerServicesImpl csImpl = (ContainerServicesImpl) cs; AcsManagerProxy acsManagerProxy = additionalContainerServices.get(cs); acsManagerProxy.shutdownNotify(); csImpl.releaseAllComponents(); ((CleaningDaemonThreadFactory) csImpl.getThreadFactory()).cleanUp(); acsManagerProxy.logoutFromManager(); additionalContainerServices.remove(cs); csImpl.cleanUp(); } catch (Throwable thr) { AcsJContainerServicesEx ex = new AcsJContainerServicesEx(); ex.setContextInfo("Failed to destroy additional container services instance"); throw ex; } } /** * Gives direct access to the CORBA ORB, POAs etc, encapsulated by the AcsCorba object. * Caution: interfering with the ComponentClient's usage of CORBA can have nasty side effects! */ public AcsCorba getAcsCorba() { return acsCorba; } /** * Gives direct access to the ACS manager, encapsulated by the AcsManagerProxy object. * Caution: interfering with ACS's usage of the manager can have nasty side effects in the entire system! */ public AcsManagerProxy getAcsManagerProxy() { return m_acsManagerProxy; } }