Package org.jboss.web.cluster

Source Code of org.jboss.web.cluster.ClusterListener$Proxy

/*
* JBoss, Home of Professional Open Source
* Copyright 2008, JBoss Inc., and individual contributors as indicated
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This 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 software 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 software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/


package org.jboss.web.cluster;


import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.net.InetAddress;
import java.net.Socket;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.StringTokenizer;

import javax.management.ObjectName;
import javax.net.ssl.KeyManagerFactory;
import javax.net.ssl.TrustManagerFactory;

import org.apache.catalina.Container;
import org.apache.catalina.ContainerEvent;
import org.apache.catalina.ContainerListener;
import org.apache.catalina.Context;
import org.apache.catalina.Engine;
import org.apache.catalina.Host;
import org.apache.catalina.Lifecycle;
import org.apache.catalina.LifecycleEvent;
import org.apache.catalina.LifecycleListener;
import org.apache.catalina.Server;
import org.apache.catalina.ServerFactory;
import org.apache.catalina.Service;
import org.apache.catalina.connector.Connector;
import org.apache.catalina.core.StandardContext;
import org.apache.catalina.core.StandardServer;
import org.apache.catalina.util.StringManager;
import org.apache.tomcat.util.IntrospectionUtils;
import org.apache.tomcat.util.buf.CharChunk;
import org.apache.tomcat.util.buf.UEncoder;
import org.apache.tomcat.util.modeler.Registry;
import org.jboss.logging.Logger;
import org.jboss.web.cluster.advertise.AdvertiseListener;



/**
* This listener communicates with a front end mod_cluster enabled proxy to
* automatically maintain the node configuration according to what is
* deployed.
*/
public class ClusterListener
    implements LifecycleListener, ContainerListener {

    protected static Logger log = Logger.getLogger(ClusterListener.class);

    /**
     * The string manager for this package.
     */
    protected StringManager sm =
        StringManager.getManager(Constants.Package);


    // -------------------------------------------------------------- Constants

   
    protected enum State { OK, ERROR, DOWN };

   
    // ----------------------------------------------------------------- Fields


    /**
     * URL encoder used to generate requests bodies.
     */
    protected UEncoder encoder = new UEncoder();
   
   
    /**
     * JMX registration information.
     */
    protected ObjectName oname;
   
   
    /**
     * Proxies.
     */
    protected Proxy[] proxies = new Proxy[0];
   
   
    /**
     * Socket factory.
     */
    protected JSSESocketFactory sslSocketFactory = null;
   
   
    /**
     * Active connections.
     */
    protected Socket[] connections = null;
   
   
    /**
     * Connection readers.
     */
    protected BufferedReader[] connectionReaders = null;
   
   
    /**
     * Connection writers.
     */
    protected BufferedWriter[] connectionWriters = null;
   
   
    /**
     * Initialization flag.
     */
    protected boolean init = false;
   
   
    /**
     * Add proxy list.
     */
    protected ArrayList<Proxy> addProxies = new ArrayList<Proxy>();
   
   
    /**
     * Remove proxy list.
     */
    protected ArrayList<Proxy> removeProxies = new ArrayList<Proxy>();
   
   
    /**
     * Advertise listener.
     */
    protected AdvertiseListener listener = null;
   
   
    // ------------------------------------------------------------- Properties


    /**
     * Receive advertisements from httpd proxies (default is to use advertisements
     * if the proxyList is not set).
     */
    protected int advertise = -1;
    public boolean getAdvertise() { return (advertise == 0) ? false : true; }
    public void setAdvertise(boolean advertise) { this.advertise = advertise ? 1 : 0; }


    /**
     * Advertise group.
     */
    protected String advertiseGroupAddress = null;
    public String getAdvertiseGroupAddress() { return advertiseGroupAddress; }
    public void setAdvertiseGroupAddress(String advertiseGroupAddress) { this.advertiseGroupAddress = advertiseGroupAddress; }


    /**
     * Advertise port.
     */
    protected int advertisePort = -1;
    public int getAdvertisePort() { return advertisePort; }
    public void setAdvertisePort(int advertisePort) { this.advertisePort = advertisePort; }


    /**
     * Advertise security key.
     */
    protected String advertiseSecurityKey = null;
    public String getAdvertiseSecurityKey() { return advertiseSecurityKey; }
    public void setAdvertiseSecurityKey(String advertiseSecurityKey) { this.advertiseSecurityKey = advertiseSecurityKey; }


    /**
     * Proxy list, format "address:port,address:port".
     */
    protected String proxyList = null;
    public String getProxyList() { return proxyList; }
    public void setProxyList(String proxyList) { this.proxyList = proxyList; }


    /**
     * URL prefix.
     */
    protected String proxyURL = null;
    public String getProxyURL() { return proxyURL; }
    public void setProxyURL(String proxyURL) { this.proxyURL = proxyURL; }


    /**
     * Connection timeout for communication with the proxy.
     */
    protected int socketTimeout = 20000;
    public int getSocketTimeout() { return socketTimeout; }
    public void setSocketTimeout(int socketTimeout) { this.socketTimeout = socketTimeout; }


    /**
     * Domain parameter, which allows tying a jvmRoute to a particular domain.
     */
    protected String domain = null;
    public String getDomain() { return domain; }
    public void setDomain(String domain) { this.domain = domain; }


    /**
     * Allows controlling flushing of packets.
     */
    protected boolean flushPackets = false;
    public boolean getFlushPackets() { return flushPackets; }
    public void setFlushPackets(boolean flushPackets) { this.flushPackets = flushPackets; }


    /**
     * Time to wait before flushing packets.
     */
    protected int flushWait = -1;
    public int getFlushWait() { return flushWait; }
    public void setFlushWait(int flushWait) { this.flushWait = flushWait; }


    /**
     * Time to wait for a pong answer to a ping.
     */
    protected int ping = -1;
    public int getPing() { return ping; }
    public void setPing(int ping) { this.ping = ping; }


    /**
     * Soft maximum inactive connection count.
     */
    protected int smax = -1;
    public int getSmax() { return smax; }
    public void setSmax(int smax) { this.smax = smax; }


    /**
     * Maximum time on seconds for idle connections above smax.
     */
    protected int ttl = -1;
    public int getTtl() { return ttl; }
    public void setTtl(int ttl) { this.ttl = ttl; }


    /**
     * Maximum time on seconds for idle connections the proxy will wait to connect to the node.
     */
    protected int nodeTimeout = -1;
    public int getNodeTimeout() { return nodeTimeout; }
    public void setNodeTimeout(int nodeTimeout) { this.nodeTimeout = nodeTimeout; }


    /**
     * Name of the balancer.
     */
    protected String balancer = null;
    public String getBalancer() { return balancer; }
    public void setBalancer(String balancer) { this.balancer = balancer; }


    /**
     * Enables sticky sessions.
     */
    protected boolean stickySession = true;
    public boolean getStickySession() { return stickySession; }
    public void setStickySession(boolean stickySession) { this.stickySession = stickySession; }


    /**
     * Remove session when the request cannot be routed to the right node.
     */
    protected boolean stickySessionRemove = false;
    public boolean getStickySessionRemove() { return stickySessionRemove; }
    public void setStickySessionRemove(boolean stickySessionRemove) { this.stickySessionRemove = stickySessionRemove; }


    /**
     * Return an error when the request cannot be routed to the right node.
     */
    protected boolean stickySessionForce = true;
    public boolean getStickySessionForce() { return stickySessionForce; }
    public void setStickySessionForce(boolean stickySessionForce) { this.stickySessionForce = stickySessionForce; }


    /**
     * Timeout to wait for an available worker (default is no wait).
     */
    protected int workerTimeout = -1;
    public int getWorkerTimeout() { return workerTimeout; }
    public void setWorkerTimeout(int workerTimeout) { this.workerTimeout = workerTimeout; }


    /**
     * Maximum number of attempts to send the request to the backend server.
     */
    protected int maxAttempts = -1;
    public int getMaxAttempts() { return maxAttempts; }
    public void setMaxAttempts(int maxAttempts) { this.maxAttempts = maxAttempts; }

   
    /**
     * SSL client cert usage to connect to the proxy.
     */
    protected boolean ssl = false;
    public boolean isSsl() { return ssl; }
    public void setSsl(boolean ssl) { this.ssl = ssl; }

   
    /**
     * SSL ciphers.
     */
    protected String sslCiphers = null;
    public String getSslCiphers() { return sslCiphers; }
    public void setSslCiphers(String sslCiphers) { this.sslCiphers = sslCiphers; }
   
   
    /**
     * SSL protocol.
     */
    protected String sslProtocol = "TLS";
    public String getSslProtocol() { return sslProtocol; }
    public void setSslProtocol(String sslProtocol) { this.sslProtocol = sslProtocol; }

   
    /**
     * Certificate encoding algorithm.
     */
    protected String sslCertificateEncodingAlgorithm = KeyManagerFactory.getDefaultAlgorithm();
    public String getSslCertificateEncodingAlgorithm() { return sslCertificateEncodingAlgorithm; }
    public void setSslCertificateEncodingAlgorithm(String sslCertificateEncodingAlgorithm) { this.sslCertificateEncodingAlgorithm = sslCertificateEncodingAlgorithm; }

   
    /**
     * SSL keystore.
     */
    protected String sslKeyStore = System.getProperty("user.home") + "/.keystore";
    public String getSslKeyStore() { return sslKeyStore; }
    public void setSslKeyStore(String sslKeyStore) { this.sslKeyStore = sslKeyStore; }
   
   
    /**
     * SSL keystore password.
     */
    protected String sslKeyStorePass = "changeit";
    public String getSslKeyStorePass() { return sslKeyStorePass; }
    public void setSslKeyStorePass(String sslKeyStorePass) { this.sslKeyStorePass = sslKeyStorePass; }
   
   
    /**
     * Keystore type.
     */
    protected String sslKeyStoreType = "JKS";
    public String getSslKeyStoreType() { return sslKeyStoreType; }
    public void setSslKeyStoreType(String sslKeyStoreType) { this.sslKeyStoreType = sslKeyStoreType; }

   
    /**
     * Keystore provider.
     */
    protected String sslKeyStoreProvider = null;
    public String getSslKeyStoreProvider() { return sslKeyStoreProvider; }
    public void setSslKeyStoreProvider(String sslKeyStoreProvider) { this.sslKeyStoreProvider = sslKeyStoreProvider; }

   
    /**
     * Truststore algorithm.
     */
    protected String sslTrustAlgorithm = TrustManagerFactory.getDefaultAlgorithm();
    public String getSslTrustAlgorithm() { return sslTrustAlgorithm; }
    public void setSslTrustAlgorithm(String sslTrustAlgorithm) { this.sslTrustAlgorithm = sslTrustAlgorithm; }

   
    /**
     * Key alias.
     */
    protected String sslKeyAlias = null;
    public String getSslKeyAlias() { return sslKeyAlias; }
    public void setSslKeyAlias(String sslKeyAlias) { this.sslKeyAlias = sslKeyAlias; }

   
    /**
     * Certificate revocation list.
     */
    protected String sslCrlFile = null;
    public String getSslCrlFile() { return sslCrlFile; }
    public void setSslCrlFile(String sslCrlFile) { this.sslCrlFile = sslCrlFile; }

   
    /**
     * Trust max certificate length.
     */
    protected int sslTrustMaxCertLength = 5;
    public int getSslTrustMaxCertLength() { return sslTrustMaxCertLength; }
    public void setSslTrustMaxCertLength(int sslTrustMaxCertLength) { this.sslTrustMaxCertLength = sslTrustMaxCertLength; }

   
    /**
     * Trust store file.
     */
    protected String sslTrustStore = System.getProperty("javax.net.ssl.trustStore");
    public String getSslTrustStore() { return sslTrustStore; }
    public void setSslTrustStore(String sslTrustStore) { this.sslTrustStore = sslTrustStore; }

   
    /**
     * Trust store password.
     */
    protected String sslTrustStorePassword = System.getProperty("javax.net.ssl.trustStorePassword");
    public String getSslTrustStorePassword() { return sslTrustStorePassword; }
    public void setSslTrustStorePassword(String sslTrustStorePassword) { this.sslTrustStorePassword = sslTrustStorePassword; }

   
    /**
     * Trust store type.
     */
    protected String sslTrustStoreType = System.getProperty("javax.net.ssl.trustStoreType");
    public String getSslTrustStoreType() { return sslTrustStoreType; }
    public void setSslTrustStoreType(String sslTrustStoreType) { this.sslTrustStoreType = sslTrustStoreType; }

   
    /**
     * Trust store provider.
     */
    protected String sslTrustStoreProvider = System.getProperty("javax.net.ssl.trustStoreProvider");
    public String getSslTrustStoreProvider() { return sslTrustStoreProvider; }
    public void setSslTrustStoreProvider(String sslTrustStoreProvider) { this.sslTrustStoreProvider = sslTrustStoreProvider; }

   
    // ---------------------------------------------- LifecycleListener Methods


    /**
     * Acknowledge the occurrence of the specified event.
     * Note: Will never be called when the listener is associated to a Server,
     * since it is not a Container.
     *
     * @param event ContainerEvent that has occurred
     */
    public void containerEvent(ContainerEvent event) {

        Container container = event.getContainer();
        Object child = event.getData();
        String type = event.getType();

        if (type.equals(Container.ADD_CHILD_EVENT)) {
            if (container instanceof Host) {
                // Deploying a webapp
                ((Lifecycle) child).addLifecycleListener(this);
                addContext((Context) child, -1);
            } else if (container instanceof Engine) {
                // Deploying a host
                container.addContainerListener(this);
            }
        } else if (type.equals(Container.REMOVE_CHILD_EVENT)) {
            if (container instanceof Host) {
                // Undeploying a webapp
                ((Lifecycle) child).removeLifecycleListener(this);
                removeContext((Context) child, -1);
            } else if (container instanceof Engine) {
                // Undeploying a host
                container.removeContainerListener(this);
            }
        }

    }


    /**
     * Primary entry point for startup and shutdown events.
     *
     * @param event The event that has occurred
     */
    public void lifecycleEvent(LifecycleEvent event) {

        Object source = event.getLifecycle();

        if (Lifecycle.START_EVENT.equals(event.getType())) {
            if (source instanceof Context) {
                // Start a webapp
                startContext((Context) source, -1);
            } else {
                return;
            }
        } else if (Lifecycle.AFTER_START_EVENT.equals(event.getType())) {
            if (source instanceof Server) {

                if (this.proxyList == null) {
                    if (advertise != 0) {
                        proxies = new Proxy[0];
                        startListener();
                    } else {
                        // Default to a httpd on localhost on the default port
                        proxies = new Proxy[1];
                        proxies[0] = new Proxy();
                    }
                } else {
                    ArrayList<Proxy> proxyList = new ArrayList<Proxy>();
                    StringTokenizer tok = new StringTokenizer(this.proxyList, ",");
                    while (tok.hasMoreTokens()) {
                        String token = tok.nextToken().trim();
                        Proxy proxy = new Proxy();
                        int pos = token.indexOf(':');
                        String address = null;
                        if (pos < 0) {
                            address = token;
                        } else if (pos == 0) {
                            address = null;
                            proxy.port = Integer.parseInt(token.substring(1));
                        } else {
                            address = token.substring(0, pos);
                            proxy.port = Integer.parseInt(token.substring(pos + 1));
                        }
                        try {
                            if (address != null) {
                                proxy.address = InetAddress.getByName(address);
                            }
                        } catch (Exception e) {
                            log.error(sm.getString("clusterListener.error.invalidHost", address), e);
                            continue;
                        }
                        proxyList.add(proxy);
                    }
                    proxies = proxyList.toArray(new Proxy[0]);
                }

                connections = new Socket[proxies.length];
                connectionReaders = new BufferedReader[proxies.length];
                connectionWriters = new BufferedWriter[proxies.length];

                sslInit();
                startServer((Server) source, -1);
               
                if (advertise == 1) {
                    startListener();
                }
               
                init = true;
            } else {
                return;
            }
        } else if (Lifecycle.STOP_EVENT.equals(event.getType())) {
            if (source instanceof Context) {
                // Stop a webapp
                stopContext((Context) source, -1);
            } else if (source instanceof Server) {
                stopListener();
                stopServer((Server) source, -1);
                for (int i = 0; i < connections.length; i++) {
                    closeConnection(i);
                }
                init = false;
            } else {
                return;
            }
        } else if (Lifecycle.PERIODIC_EVENT.equals(event.getType())) {
            if (init && source instanceof Engine) {
                status((Engine) source);
            }
        }

    }
   
   
    /**
     * Add proxy.
     */
    public void addProxy(String address) {
        int pos = address.indexOf(':');
        String host = null;
        int port = 0;
        if (pos < 0) {
            host = address;
        } else if (pos == 0) {
            host = null;
            port = Integer.parseInt(address.substring(1));
        } else {
            host = address.substring(0, pos);
            port = Integer.parseInt(address.substring(pos + 1));
        }
        addProxy(host, port);
    }
   
   
    /**
     * Add proxy.
     */
    public synchronized void addProxy(String host, int port) {
        Proxy proxy = new Proxy();
        try {
            proxy.address = InetAddress.getByName(host);
        } catch (Exception e) {
            throw new IllegalArgumentException(e);
        }
        if (port > 0) {
            proxy.port = port;
        }
        proxy.state = State.ERROR;
        addProxies.add(proxy);
    }
   
   
    /**
     * Remove proxy.
     */
    public synchronized void removeProxy(String host, int port) {
        Proxy proxy = new Proxy();
        try {
            proxy.address = InetAddress.getByName(host);
        } catch (Exception e) {
            throw new IllegalArgumentException(e);
        }
        if (port > 0) {
            proxy.port = port;
        }
        removeProxies.add(proxy);
    }
   
   
    /**
     * Retrieves the full proxy configuration. To be used through JMX or similar.
     *
     *         response: HTTP/1.1 200 OK
     *   response:
     *   node: [1:1] JVMRoute: node1 Domain: [bla] Host: 127.0.0.1 Port: 8009 Type: ajp
     *   host: 1 [] vhost: 1 node: 1
     *   context: 1 [/] vhost: 1 node: 1 status: 1
     *   context: 2 [/myapp] vhost: 1 node: 1 status: 1
     *   context: 3 [/host-manager] vhost: 1 node: 1 status: 1
     *   context: 4 [/docs] vhost: 1 node: 1 status: 1
     *   context: 5 [/manager] vhost: 1 node: 1 status: 1
     *
     * @return the proxy confguration
     */
    public String getProxyConfiguration() {
        HashMap<String, String> parameters = new HashMap<String, String>();
        // Send DUMP * request
      Proxy[] local = proxies;
      StringBuffer result = new StringBuffer();
      for (int i = 0; i < local.length; i++) {
        result.append("Proxy[").append(i).append("]: [").append(local[i].address)
            .append(':').append(local[i].port).append("]: \r\n");
        result.append(sendRequest("DUMP", true, parameters, i));
        result.append("\r\n");
      }
      return result.toString();
    }
    /**
     * Retrieves the full proxy info message.
     *
     *
     * @return the proxy confguration
     */
    public String getProxyInfo() {
        HashMap<String, String> parameters = new HashMap<String, String>();
        // Send INFO * request
      Proxy[] local = proxies;
      StringBuffer result = new StringBuffer();
      for (int i = 0; i < local.length; i++) {
        result.append("Proxy[").append(i).append("]: [").append(local[i].address)
            .append(':').append(local[i].port).append("]: \r\n");
        result.append(sendRequest("INFO", true, parameters, i));
        result.append("\r\n");
      }
      return result.toString();
    }
    /**
     * Check the node connectivity with the proxies.
     *
     *
     * @return the status of the node or the proxy.
     */
    public String doProxyPing(String JvmRoute) {
        HashMap<String, String> parameters = new HashMap<String, String>();
        if (JvmRoute != null)
            parameters.put("JVMRoute", JvmRoute);
        // Send PING * request
        Proxy[] local = proxies;
        StringBuffer result = new StringBuffer();
        for (int i = 0; i < local.length; i++) {
            result.append("Proxy[").append(i).append("]: [").append(local[i].address)
                            .append(':').append(local[i].port).append("]: \r\n");
            result.append(sendRequest("PING", true, parameters, i));
            result.append("\r\n");
        }
        return result.toString();
    }  
    public String doProxyPing(String scheme, String host, int port) {
        return doProxyPing(scheme + "://" + host + String.valueOf(port));
    }
   
    /**
     * Reset a DOWN connection to the proxy up to ERROR, where the configuration will
     * be refreshed. To be used through JMX or similar.
     */
    public void reset() {
      Proxy[] local = proxies;
      for (int i = 0; i < local.length; i++) {
        if (local[i].state == State.DOWN) {
          local[i].state = State.ERROR;
        }
      }
    }
   
   
    /**
     * Refresh configuration. To be used through JMX or similar.
     */
    public void refresh() {
        // Set as error, and the periodic event will refresh the configuration
      Proxy[] local = proxies;
      for (int i = 0; i < local.length; i++) {
        if (local[i].state == State.OK) {
          local[i].state = State.ERROR;
        }
      }
    }
   
   
    /**
     * Disable all webapps for all engines. To be used through JMX or similar.
     */
    public boolean disable() {
      Service[] services = ServerFactory.getServer().findServices();
        for (int i = 0; i < services.length; i++) {
            Engine engine = (Engine) services[i].getContainer();
            HashMap<String, String> parameters = new HashMap<String, String>();
            parameters.put("JVMRoute", engine.getJvmRoute());
            // Send DISABLE-APP * request
            sendRequest("DISABLE-APP", true, parameters);
        }
        return (proxies[0].state == State.OK);
    }
   
   
    /**
     * Enable all webapps for all engines. To be used through JMX or similar.
     */
    public boolean enable() {
      Service[] services = ServerFactory.getServer().findServices();
        for (int i = 0; i < services.length; i++) {
            Engine engine = (Engine) services[i].getContainer();
            HashMap<String, String> parameters = new HashMap<String, String>();
            parameters.put("JVMRoute", engine.getJvmRoute());
            // Send ENABLE-APP * request
            sendRequest("ENABLE-APP", true, parameters);
        }
        return (proxies[0].state == State.OK);
    }
   
   
    /**
     * Start the advertise listener.
     */
    protected void startListener() {
        listener = new AdvertiseListener(this);
        if (advertiseGroupAddress != null) {
            listener.setGroupAddress(advertiseGroupAddress);
        }
        if (advertisePort > 0) {
            listener.setAdvertisePort(advertisePort);
        }
        try {
            if (advertiseSecurityKey != null) {
                listener.setSecurityKey(advertiseSecurityKey);
            }
            listener.start();
        } catch (IOException e) {
            log.error(sm.getString("clusterListener.error.startListener"), e);
        } catch (NoSuchAlgorithmException e) {
            // Should never happen
            log.error(sm.getString("clusterListener.error.startListener"), e);
        }
    }


    /**
     * Stop the advertise listener.
     */
    protected void stopListener() {
        if (listener != null) {
            try {
                listener.destroy();
            } catch (IOException e) {
                log.error(sm.getString("clusterListener.error.stopListener"), e);
            }
            listener = null;
        }
    }


    /**
     * Send commands to the front end server assocaited with the startup of the
     * node.
     */
    protected void startServer(Server server, int pos) {

        // JMX registration
        if (oname==null) {
            try {
                oname = new ObjectName(((StandardServer) server).getDomain() + ":type=ClusterListener");
                Registry.getRegistry(null, null).registerComponent(this, oname, null);
            } catch (Exception e) {
                log.error(sm.getString("clusterListener.error.jmxRegister"), e);
            }
        }
               
        Service[] services = server.findServices();
        for (int i = 0; i < services.length; i++) {
            services[i].getContainer().addContainerListener(this);

            Engine engine = (Engine) services[i].getContainer();
            ((Lifecycle) engine).addLifecycleListener(this);
            Connector connector = findProxyConnector(engine.getService().findConnectors());
            InetAddress localAddress =
                (InetAddress) IntrospectionUtils.getProperty(connector.getProtocolHandler(), "address");
            if ((engine.getJvmRoute() == null || localAddress == null) && proxies.length > 0) {
                // Automagical JVM route (address + port + engineName)
                try {
                    if (localAddress == null) {
                        Socket connection = getConnection(0);
                        localAddress = connection.getLocalAddress();
                        if (localAddress != null) {
                            IntrospectionUtils.setProperty(connector.getProtocolHandler(), "address", localAddress.getHostAddress());
                        } else {
                            // Should not happen
                            IntrospectionUtils.setProperty(connector.getProtocolHandler(), "address", "127.0.0.1");
                        }
                        log.info(sm.getString("clusterListener.address", localAddress.getHostAddress()));
                    }
                    if (engine.getJvmRoute() == null) {
                        String hostName = null;
                        if (localAddress != null) {
                            hostName = localAddress.getHostName();
                        } else {
                            // Fallback
                            hostName = "127.0.0.1";
                        }
                        String jvmRoute = hostName + ":" + connector.getPort() + ":" + engine.getName();
                        engine.setJvmRoute(jvmRoute);
                        log.info(sm.getString("clusterListener.jvmRoute", engine.getName(), jvmRoute));
                    }
                } catch (Exception e) {
                    proxies[0].state = State.ERROR;
                    log.info(sm.getString("clusterListener.error.addressJvmRoute"), e);
                    return;
                }
            }
           
            config(engine, pos);
            Container[] children = engine.findChildren();
            for (int j = 0; j < children.length; j++) {
                children[j].addContainerListener(this);
                Container[] children2 = children[j].findChildren();
                for (int k = 0; k < children2.length; k++) {
                    ((Lifecycle) children2[k]).addLifecycleListener(this);
                    addContext((Context) children2[k], pos);
                }
            }
        }
    }
   
   
    /**
     * Send commands to the front end server associated with the shutdown of the
     * node.
     */
    protected void stopServer(Server server, int pos) {
       
        // JMX unregistration
        if (oname==null) {
            try {
                Registry.getRegistry(null, null).unregisterComponent(oname);
            } catch (Exception e) {
                log.error(sm.getString("clusterListener.error.jmxUnregister"), e);
            }
        }

        Service[] services = server.findServices();
        for (int i = 0; i < services.length; i++) {
            services[i].getContainer().removeContainerListener(this);
            ((Lifecycle) services[i].getContainer()).removeLifecycleListener(this);
            removeAll((Engine) services[i].getContainer(), pos);
            Container[] children = services[i].getContainer().findChildren();
            for (int j = 0; j < children.length; j++) {
                children[j].removeContainerListener(this);
                Container[] children2 = children[j].findChildren();
                for (int k = 0; k < children2.length; k++) {
                    ((Lifecycle) children2[k]).removeLifecycleListener(this);
                    removeContext((Context) children2[k], pos);
                }
            }
        }
    }
   

    /**
     * Reset configuration for a particular proxy following an error.
     */
    protected void reset(int pos) {
      
        Service[] services = ServerFactory.getServer().findServices();
        for (int i = 0; i < services.length; i++) {
            Engine engine = (Engine) services[i].getContainer();
            removeAll((Engine) services[i].getContainer(), pos);
            config(engine, pos);
            Container[] children = engine.findChildren();
            for (int j = 0; j < children.length; j++) {
                Container[] children2 = children[j].findChildren();
                for (int k = 0; k < children2.length; k++) {
                    addContext((Context) children2[k], pos);
                }
            }
        }
       
    }
   

    /**
     * Send the configuration for the specified engine to the proxy.
     *
     * @param engine
     */
    protected void config(Engine engine, int pos) {
        if (log.isDebugEnabled()) {
            log.debug(sm.getString("clusterListener.config", engine.getName()));
        }
        // Collect configuration from the connectors and service and call CONFIG
        Connector connector = findProxyConnector(engine.getService().findConnectors());
        HashMap<String, String> parameters = new HashMap<String, String>();
        parameters.put("JVMRoute", engine.getJvmRoute());
        boolean reverseConnection =
            Boolean.TRUE.equals(IntrospectionUtils.getProperty(connector.getProtocolHandler(), "reverseConnection"));
        boolean ssl =
            Boolean.TRUE.equals(IntrospectionUtils.getProperty(connector.getProtocolHandler(), "SSLEnabled"));
        boolean ajp = ((String) IntrospectionUtils.getProperty(connector.getProtocolHandler(), "name")).startsWith("ajp-");

        if (reverseConnection) {
            parameters.put("Reversed", "true");
        }
        parameters.put("Host", getAddress(connector));
        parameters.put("Port", "" + connector.getPort());
        if (ajp) {
            parameters.put("Type", "ajp");
        } else if (ssl) {
            parameters.put("Type", "https");
        } else {
            parameters.put("Type", "http");
        }
       
        // Other configuration parameters
        if (domain != null) {
          parameters.put("Domain", domain);
        }
        if (flushPackets) {
          parameters.put("flushpackets", "On");
        }
        if (flushWait != -1) {
          parameters.put("flushwait", "" + flushWait);
        }
        if (ping != -1) {
          parameters.put("ping", "" + ping);
        }
        if (smax != -1) {
          parameters.put("smax", "" + smax);
        }
        if (ttl != -1) {
          parameters.put("ttl", "" + ttl);
        }
        if (nodeTimeout != -1) {
          parameters.put("Timeout", "" + nodeTimeout);
        }
        if (balancer != null) {
          parameters.put("Balancer", balancer);
        }
        if (!stickySession) {
          parameters.put("StickySession", "No");
        }
        if (!org.apache.catalina.Globals.SESSION_COOKIE_NAME.equals("JSESSIONID")) {
          parameters.put("StickySessionCookie", org.apache.catalina.Globals.SESSION_COOKIE_NAME);
        }
        if (!org.apache.catalina.Globals.SESSION_PARAMETER_NAME.equals("jsessionid")) {
          parameters.put("StickySessionPath", org.apache.catalina.Globals.SESSION_PARAMETER_NAME);
        }
        if (stickySessionRemove) {
          parameters.put("StickSessionRemove", "Yes");
        }
        if (!stickySessionForce) {
          parameters.put("StickySessionForce", "No");
        }
        if (workerTimeout != -1) {
          parameters.put("WaitWorker", "" + workerTimeout);
        }
        if (maxAttempts != -1) {
          parameters.put("Maxattempts", "" + maxAttempts);
        }
       
        // Send CONFIG request
        sendRequest("CONFIG", false, parameters, pos);
    }


    /**
     * Remove all contexts from the specified engine.
     *
     * @param engine
     */
    protected void removeAll(Engine engine, int pos) {
        if (log.isDebugEnabled()) {
            log.debug(sm.getString("clusterListener.stop", engine.getName()));
        }

        // JVMRoute can be null here if nothing was ever initialized
        if (engine.getJvmRoute() != null) {
            HashMap<String, String> parameters = new HashMap<String, String>();
            parameters.put("JVMRoute", engine.getJvmRoute());

            // Send REMOVE-APP * request
            sendRequest("REMOVE-APP", true, parameters, pos);
        }
    }


    /**
     * Send a periodic status request. If in error state, the listener will attempt to refresh
     * the configuration on the front end server.
     *
     * @param engine
     */
    protected synchronized void status(Engine engine) {
       
        // Check to add or remove proxies, and rebuild a new list if needed
        if (!addProxies.isEmpty() || !removeProxies.isEmpty()) {
            ArrayList<Proxy> currentProxies = new ArrayList<Proxy>();
            for (int i = 0; i < proxies.length; i++) {
                currentProxies.add(proxies[i]);
            }
            for (int i = 0; i < addProxies.size(); i++) {
                if (!currentProxies.contains(addProxies.get(i))) {
                    currentProxies.add(addProxies.get(i));
                }
            }
            for (int i = 0; i < removeProxies.size(); i++) {
                if (currentProxies.contains(removeProxies.get(i))) {
                    currentProxies.remove(removeProxies.get(i));
                }
            }
            addProxies.clear();
            removeProxies.clear();
            proxies = currentProxies.toArray(new Proxy[0]);
            // Reset all connections
            if (connections != null) {
                for (int i = 0; i < connections.length; i++) {
                    closeConnection(i);
                }
            }
            connections = new Socket[proxies.length];
            connectionReaders = new BufferedReader[proxies.length];
            connectionWriters = new BufferedWriter[proxies.length];
        }
       
        Proxy[] local = proxies;
        for (int i = 0; i < local.length; i++) {
            if (local[i].state == State.ERROR) {
                local[i].state = State.OK;
                // Something went wrong in a status at some point, so fully restore the configuration
                reset(i);
            } else if (local[i].state == State.OK) {
                if (log.isDebugEnabled()) {
                    log.debug(sm.getString("clusterListener.status", engine.getName()));
                }
                HashMap<String, String> parameters = new HashMap<String, String>();
                parameters.put("JVMRoute", engine.getJvmRoute());
                parameters.put("Load", "1");
                // Send STATUS request
                sendRequest("STATUS", false, parameters, i);
            }
        }
    }

   
    /**
     * Add a new context.
     *
     * @param context
     */
    protected void addContext(Context context, int pos) {
        if (log.isDebugEnabled()) {
            log.debug(sm.getString("clusterListener.context.enable", context.getPath(), context.getParent().getName(), ((StandardContext) context).getState()));
        }

        HashMap<String, String> parameters = new HashMap<String, String>();
        parameters.put("JVMRoute", getJvmRoute(context));
        parameters.put("Context", ("".equals(context.getPath())) ? "/" : context.getPath());
        parameters.put("Alias", getHost(context));

        // Send ENABLE-APP if state is started
        if (context.isStarted()) {
            sendRequest("ENABLE-APP", false, parameters, pos);
        }
    }


    /**
     * Remove a context.
     *
     * @param context
     */
    protected void removeContext(Context context, int pos) {
        if (log.isDebugEnabled()) {
            log.debug(sm.getString("clusterListener.context.disable", context.getPath(), context.getParent().getName(), ((StandardContext) context).getState()));
        }

        HashMap<String, String> parameters = new HashMap<String, String>();
        String jvmRoute = getJvmRoute(context);
        // JVMRoute can be null here if nothing was ever initialized
        if (jvmRoute != null) {
            parameters.put("JVMRoute", jvmRoute);
            parameters.put("Context", ("".equals(context.getPath())) ? "/" : context.getPath());
            parameters.put("Alias", getHost(context));

            // Send REMOVE-APP
            sendRequest("REMOVE-APP", false, parameters, pos);
        }
    }


    /**
     * Start a context.
     *
     * @param context
     */
    protected void startContext(Context context, int pos) {
        if (log.isDebugEnabled()) {
            log.debug(sm.getString("clusterListener.context.start", context.getPath(), context.getParent().getName()));
        }

        HashMap<String, String> parameters = new HashMap<String, String>();
        parameters.put("JVMRoute", getJvmRoute(context));
        parameters.put("Context", ("".equals(context.getPath())) ? "/" : context.getPath());
        parameters.put("Alias", getHost(context));

        // Send ENABLE-APP
        sendRequest("ENABLE-APP", false, parameters, pos);
    }


    /**
     * Stop a context.
     *
     * @param context
     */
    protected void stopContext(Context context, int pos) {
        if (log.isDebugEnabled()) {
            log.debug(sm.getString("clusterListener.context.stop", context.getPath(), context.getParent().getName()));
        }

        HashMap<String, String> parameters = new HashMap<String, String>();
        parameters.put("JVMRoute", getJvmRoute(context));
        parameters.put("Context", ("".equals(context.getPath())) ? "/" : context.getPath());
        parameters.put("Alias", getHost(context));

        // Send STOP-APP
        sendRequest("STOP-APP", false, parameters, pos);
    }

   
    /**
     * Return the JvmRoute for the specified context.
     *
     * @param context
     * @return
     */
    protected String getJvmRoute(Context context) {
        return ((Engine) context.getParent().getParent()).getJvmRoute();
    }
   

    /**
     * Return the host and its alias list with which the context is associated.
     *
     * @param context
     * @return
     */
    protected String getHost(Context context) {
        StringBuffer result = new StringBuffer();
        Host host = (Host) context.getParent();
        result.append(host.getName());
        String[] aliases = host.findAliases();
        for (int i = 0; i < aliases.length; i++) {
            result.append(',');
            result.append(aliases[i]);
        }
        return result.toString();
    }
   
   
    /**
     * Find the most likely connector the proxy server should connect to, or
     * accept connections from.
     *
     * @param connectors
     * @return
     */
    protected Connector findProxyConnector(Connector[] connectors) {
        int pos = 0;
        int maxThreads = 0;
        for (int i = 0; i < connectors.length; i++) {
            if (connectors[i].getProtocol().startsWith("AJP")) {
                // Return any AJP connector found
                return connectors[i];
            }
            if (Boolean.TRUE.equals(IntrospectionUtils.getProperty(connectors[i].getProtocolHandler(), "reverseConnection"))) {
                return connectors[i];
            }
            Integer mt = (Integer) IntrospectionUtils.getProperty(connectors[i].getProtocolHandler(), "maxThreads");
            if (mt.intValue() > maxThreads) {
                maxThreads = mt.intValue();
                pos = i;
            }
        }
        // If no AJP connector and no reverse, return the connector with the most threads
        return connectors[pos];
    }

   
    /**
     * Return the address on which the connector is bound.
     *
     * @param connector
     * @return
     */
    protected String getAddress(Connector connector) {
        InetAddress inetAddress =
            (InetAddress) IntrospectionUtils.getProperty(connector.getProtocolHandler(), "address");
        if (inetAddress == null) {
            // Should not happen
            return "127.0.0.1";
        } else {
            return inetAddress.getHostAddress();
        }
    }
   
   
    /**
     * Send HTTP request, with the specified list of parameters. If an IO error occurs, the error state will
     * be set. If the front end server reports an error, will mark as error state. Other unexpected exceptions
     * will be thrown and the error state will be set.
     *
     * @param command
     * @param wildcard
     * @param parameters
     * @return the response body as a String; null if in error state or a normal error occurs
     */
    protected void sendRequest(String command, boolean wildcard, HashMap<String, String> parameters) {
      sendRequest(command, wildcard, parameters, -1);
    }

    protected synchronized String sendRequest(String command, boolean wildcard, HashMap<String, String> parameters, int pos) {

        BufferedReader reader = null;
        BufferedWriter writer = null;
        CharChunk body = null;
       
        // First, encode the POST body
        try {
            body = encoder.encodeURL("", 0, 0);
            body.recycle();
            Iterator<String> keys = parameters.keySet().iterator();
            while (keys.hasNext()) {
                String key = keys.next();
                String value = parameters.get(key);
                if (value == null) {
                    throw new IllegalArgumentException(sm.getString("clusterListener.error.nullAttribute", key));
                }
                body = encoder.encodeURL(key, 0, key.length());
                body.append('=');
                if (value != null) {
                    body = encoder.encodeURL(value, 0, value.length());
                }
                if (keys.hasNext()) {
                    body.append('&');
                }
            }
        } catch (IOException e) {
            body.recycle();
            // Error encoding URL, should not happen
            throw new IllegalArgumentException(e);
        }

        int start = 0;
        int end = proxies.length;
        if (pos != -1) {
            start = pos;
            end = pos + 1;
        }

        for (int i = start; i < end; i++) {

            // If there was an error, do nothing until the next periodic event, where the whole configuration
            // will be refreshed
            if (proxies[i].state != State.OK) {
                continue;
            }
           
            if (log.isDebugEnabled()) {
                log.debug(sm.getString("clusterListener.request", command, wildcard, proxies[i]));
            }
           
            try {

                // Then, connect to the proxy
                getConnection(i);
                writer = getConnectionWriter(i);
                // Check connection to see if it is still alive (not really allowed,
                // but httpd deals with the extra CRLF)
                try {
                    writer.write("\r\n");
                    writer.flush();
                } catch (IOException e) {
                    // Get a new connection; if it fails this second time, it is an error
                    closeConnection(i);
                    getConnection(i);
                    writer = getConnectionWriter(i);
                }

                // Generate and write request
                String url = proxyURL;
                if (url == null) {
                    url = (wildcard) ? "/*" : "/";
                } else {
                    if (url.endsWith("/") && wildcard) {
                        url = url + "*";
                    } else if (wildcard) {
                        url = url + "/*";
                    }
                }
                String requestLine = command + " " + url + " HTTP/1.0";
                writer.write(requestLine);
                writer.write("\r\n");
                writer.write("Content-Length: " + body.getLength() + "\r\n");
                writer.write("User-Agent: ClusterListener/1.0\r\n");
                writer.write("Connection: Keep-Alive\r\n");
                writer.write("\r\n");
                writer.write(body.getBuffer(), body.getStart(), body.getLength());
                writer.write("\r\n");
                writer.flush();

                // Read the response to a string
                reader = getConnectionReader(i);
                // Read the first response line and skip the rest of the HTTP header
                String responseStatus = reader.readLine();
                // Parse the line, which is formed like HTTP/1.x YYY Message
                int status = 500;
                String version = "0";
                String message = null;
                String errorType = null;
                int contentLength = 0;
                if (responseStatus != null) {
                    try {
                        responseStatus = responseStatus.substring(responseStatus.indexOf(' ') + 1, responseStatus.indexOf(' ', responseStatus.indexOf(' ') + 1));
                        status = Integer.parseInt(responseStatus);
                        String header = reader.readLine();
                        while (!"".equals(header)) {
                            int colon = header.indexOf(':');
                            String headerName = header.substring(0, colon).trim();
                            String headerValue = header.substring(colon + 1).trim();
                            if ("version".equalsIgnoreCase(headerName)) {
                                version = headerValue;
                            } else if ("type".equalsIgnoreCase(headerName)) {
                                errorType = headerValue;
                            } else if ("mess".equalsIgnoreCase(headerName)) {
                                message = headerValue;
                            } else if ("content-length".equalsIgnoreCase(headerName)) {
                                contentLength = Integer.parseInt(headerValue);
                            }
                            header = reader.readLine();
                        }
                    } catch (Exception e) {
                        log.info(sm.getString("clusterListener.error.parse", command), e);
                    }
                }

                // Mark as error if the front end server did not return 200; the configuration will
                // be refreshed during the next periodic event
                if (status == 200) {
                    // Read the request body
                    StringBuffer result = new StringBuffer();
                    if (contentLength > 0) {
                        int thisTime = contentLength;
                        char[] buf = new char[512];
                        while (contentLength > 0) {
                            thisTime = (contentLength > buf.length) ? buf.length : contentLength;
                            int n = reader.read(buf, 0, thisTime);
                            if (n <= 0) {
                                break;
                            } else {
                                result.append(buf, 0, n);
                                contentLength -= n;
                            }
                        }
                    }
                    if (pos != -1) {
                        return result.toString();
                    }
                } else {
                    if ("SYNTAX".equals(errorType)) {
                        // Syntax error means the protocol is incorrect, which cannot be automatically fixed
                        proxies[i].state = State.DOWN;
                        log.error(sm.getString("clusterListener.error.syntax", command, proxies[i], errorType, message));
                    } else {
                        proxies[i].state = State.ERROR;
                        log.error(sm.getString("clusterListener.error.other", command, proxies[i], errorType, message));
                    }
                }

            } catch (IOException e) {
                // Most likely this is a connection error with the proxy
                proxies[i].state = State.ERROR;
                log.info(sm.getString("clusterListener.error.io", command, proxies[i]), e);
            } finally {
                // If there's an error of any sort, or if the proxy did not return 200, it is an error
                if (proxies[i].state != State.OK) {
                    closeConnection(i);
                }
            }

        }

        return null;
       
    }

   
    /**
     * SSL init.
     */
    protected void sslInit() {
        if (ssl) {
            sslSocketFactory = new JSSESocketFactory(this);
        }
    }
   
   
    /**
     * Return a reader to the proxy.
     */
    protected Socket getConnection(int i)
        throws IOException {
        if (connections[i] == null) {
            InetAddress address = (proxies[i].address == null) ? InetAddress.getLocalHost() : proxies[i].address;
            if (ssl) {
                connections[i] = sslSocketFactory.createSocket(address, proxies[i].port);
            } else {
                connections[i] = new Socket(address, proxies[i].port);
            }
            connections[i].setSoTimeout(socketTimeout);
        }
        return connections[i];
    }
   

    /**
     * Return a reader to the proxy.
     */
    protected BufferedReader getConnectionReader(int i)
        throws IOException {
        if (connectionReaders[i] == null) {
            connectionReaders[i] = new BufferedReader(new InputStreamReader(connections[i].getInputStream()));
        }
        return connectionReaders[i];
    }
   

    /**
     * Return a writer to the proxy.
     */
    protected BufferedWriter getConnectionWriter(int i)
        throws IOException {
        if (connectionWriters[i] == null) {
            connectionWriters[i] = new BufferedWriter(new OutputStreamWriter(connections[i].getOutputStream()));
        }
        return connectionWriters[i];
    }
   

    /**
     * Close connection.
     */
    protected void closeConnection(int i) {
        try {
            if (connectionReaders[i] != null) {
                connectionReaders[i].close();
            }
        } catch (IOException e) {
            // Ignore
        }
        connectionReaders[i] = null;
        try {
            if (connectionWriters[i] != null) {
                connectionWriters[i].close();
            }
        } catch (IOException e) {
            // Ignore
        }
        connectionWriters[i] = null;
        try {
            if (connections[i] != null) {
                connections[i].close();
            }
        } catch (IOException e) {
            // Ignore
        }
        connections[i] = null;
    }
   
   
    // ------------------------------------------------------ Proxy Inner Class

   
    /**
     * This class represents a front-end httpd server.
     */
    protected static class Proxy {
      public InetAddress address = null;
      public int port = 8000;
      public State state = State.OK;
     
      public String toString() {
          if (address == null) {
              return ":" + port;
          } else {
              return address.getHostAddress() + ":" + port;
          }
      }
     
      public boolean equals(Object o) {
          if (o instanceof Proxy) {
              Proxy compare = (Proxy) o;
              if (port != compare.port) {
                  return false;
              }
              if (compare.address == null) {
                  if (address == null) {
                      return true;
                  }
              } else if ((compare.address.equals(address)) && port == compare.port) {
                  return true;
              }
          }
          return false;
      }
     
    }
   
   
}
TOP

Related Classes of org.jboss.web.cluster.ClusterListener$Proxy

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.