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

org.fusesource.fabric.bridge.GatewayConnector Maven / Gradle / Ivy

/**
 * Copyright (C) FuseSource, Inc.
 * http://fusesource.com
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.fusesource.fabric.bridge;

import org.fusesource.fabric.bridge.internal.AbstractConnector;
import org.fusesource.fabric.bridge.internal.SourceConnector;
import org.fusesource.fabric.bridge.internal.TargetConnector;
import org.fusesource.fabric.bridge.model.BridgeDestinationsConfig;
import org.fusesource.fabric.bridge.model.BridgedDestination;
import org.fusesource.fabric.bridge.model.BrokerConfig;
import org.fusesource.fabric.bridge.model.RemoteBridge;
import org.springframework.jms.JmsException;
import org.springframework.jms.UncategorizedJmsException;

import javax.xml.bind.annotation.*;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

/**
 * Connects {@link BridgeConnector BridgeConnectors} on remote brokers to a
 * {@link TargetConnector} on local broker. Also creates {@link SourceConnector
 * SourceConnectors} for every {@link BridgeConnector} based on its
 * {@link RemoteBridge}.
 * 
 * @author Dhiraj Bokde
 * 
 */
@XmlRootElement(name="gateway-connector")
@XmlAccessorType(XmlAccessType.NONE)
@XmlType(propOrder = {"localBrokerConfig", "inboundDestinations", "outboundDestinations", "remoteBridges"})
public class GatewayConnector extends AbstractConnector {

	@XmlElement(name="local-broker", required=true)
	private BrokerConfig localBrokerConfig;

	@XmlElement(name="inbound-destinations")
	private BridgeDestinationsConfig inboundDestinations;
	
	// shared by all bridge connectors connected to this gateway
	@XmlElement(name="outbound-destinations")
	private BridgeDestinationsConfig outboundDestinations;

    @XmlAttribute
    private String inboundDestinationsRef;

    // shared by all bridge connectors connected to this gateway
    @XmlAttribute
    private String outboundDestinationsRef;

	// default is an empty list of brokers to be added later using addBrokerConfig
	@XmlElement(name="remote-bridge")
	private List remoteBridges = new ArrayList();

	private final Map outboundConnectors = new HashMap();
	
	private TargetConnector defaultInboundConnector;

    private final Map inboundConnectors = new HashMap();

	@Override
	public void afterPropertiesSet() throws Exception {
		if (localBrokerConfig == null) {
			throw new IllegalArgumentException("Property local-broker must be set");
		}
		if (inboundDestinations == null && outboundDestinations == null) {
			throw new IllegalArgumentException(
                "At least one of inbound-destinations or outbound-destinations must be set");
		}
	}

    @Override
	protected void doInitialize() {
		// create the inbound connector shared by all bridgeconnectors connected to this gateway
		if (inboundDestinations != null && inboundDestinations.isUseStagingQueue()) {
			createDefaultInboundConnector();
        } else if (inboundDestinations != null && !inboundDestinations.isUseStagingQueue()) {
            LOG.info("Inbound connector NOT created for default inbound destinations since staging queue is NOT used");
		} else {
            LOG.info("Inbound connector NOT created since default inbound destinations are NOT specified");
        }
		
		// create outbound connectors for outboundBrokers
		if (remoteBridges != null && !remoteBridges.isEmpty()) {
			createRemoteBridgeConnectors();
		}

		LOG.info("Initialized");
	}

	private void createDefaultInboundConnector() {
		defaultInboundConnector = new TargetConnector();
		defaultInboundConnector.setAutoStartup(isAutoStartup());
		defaultInboundConnector.setPhase(getPhase());
	
		defaultInboundConnector.setLocalBrokerConfig(localBrokerConfig);
		defaultInboundConnector.setInboundDestinations(inboundDestinations);
		try {
			defaultInboundConnector.afterPropertiesSet();
		} catch (Exception e) {
			String msg = "Error creating inbound connector: " + e.getMessage();
			LOG.error(msg, e);
			throw new IllegalArgumentException(msg, e);
		}
	}

	private void createRemoteBridgeConnectors() {
		for (RemoteBridge remoteBridge : remoteBridges) {

            // does the remote bridge use custom inbound destinations??
            // TODO maybe this should only check for staging queue name or location
            if (remoteBridge.getInboundDestinations() != null &&
                !remoteBridge.getInboundDestinations().equals(inboundDestinations)
                && remoteBridge.getInboundDestinations().isUseStagingQueue()) {
                TargetConnector inboundConnector = createInboundConnector(remoteBridge);
                inboundConnectors.put(remoteBridge, inboundConnector);
            } else {
                String reason;
                if (inboundDestinations == null) {
                    reason = ", since it is Unidirectional";
                } else {
                    reason = inboundDestinations.isUseStagingQueue() ?
                        ", since it uses default inbound connector" : ", since staging queue is NOT used";
                }
                LOG.warn("Remote bridge " + remoteBridge + " does NOT require inbound connector" +
                    reason);
            }

            if ((remoteBridge.getOutboundDestinations() != null) ||
                (this.outboundDestinations != null)) {
                SourceConnector outboundConnector = createOutboundConnector(remoteBridge);
                outboundConnectors.put(remoteBridge, outboundConnector);
            }

		}
	}

    private TargetConnector createInboundConnector(RemoteBridge remoteBridge) {
        TargetConnector inboundConnector = new TargetConnector();
        inboundConnector.setId(getId() + "." + remoteBridge.getId() + ".inboundConnector");
        inboundConnector.setAutoStartup(isAutoStartup());
        inboundConnector.setPhase(getPhase());

        inboundConnector.setLocalBrokerConfig(localBrokerConfig);
        // check if the remote broker config should be used
        if (!remoteBridge.getInboundDestinations().isDefaultStagingLocation()) {
            inboundConnector.setRemoteBrokerConfig(remoteBridge.getRemoteBrokerConfig());
        }
        inboundConnector.setInboundDestinations(remoteBridge.getInboundDestinations());

        try {
            inboundConnector.afterPropertiesSet();
        } catch (Exception e) {
            String msg = "Error creating inbound connector for " + remoteBridge + " : " + e.getMessage();
            LOG.error(msg, e);
            throw new IllegalArgumentException(msg, e);
        }
        return inboundConnector;
    }

    private SourceConnector createOutboundConnector(
			RemoteBridge remoteBridge) {
		SourceConnector outboundConnector = new SourceConnector();
        outboundConnector.setId(getId() + "." + remoteBridge.getId() + ".outboundConnector");
		outboundConnector.setAutoStartup(isAutoStartup());
		outboundConnector.setPhase(getPhase());
		
		outboundConnector.setLocalBrokerConfig(localBrokerConfig);
		outboundConnector.setRemoteBrokerConfig(remoteBridge.getRemoteBrokerConfig());
		if (remoteBridge.getOutboundDestinations() == null) {
			// use default destinations
            LOG.warn("Remote bridge " + remoteBridge + " does not specify outbound destinations" +
                (outboundDestinations == null ? ", it is unidirectional!" :
                    ", will load balance messages from default outbound destinations!"));
			outboundConnector.setOutboundDestinations(outboundDestinations);
		} else {
			// completely override outbound destinations for this bridge
			outboundConnector.setOutboundDestinations(remoteBridge.getOutboundDestinations());
		}
		
		try {
			outboundConnector.afterPropertiesSet();
		} catch (Exception e) {
			String msg = "Error creating outbound connector for " + remoteBridge + " : " + e.getMessage();
			LOG.error(msg, e);
			throw new IllegalArgumentException(msg, e);
		}
		return outboundConnector;
	}

	@Override
	protected void doStart() {
		if (defaultInboundConnector != null) {
			// start the shared inbound connector
			try {
				defaultInboundConnector.start();
			} catch (Exception e) {
				String msg = "Error starting inbound connector : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}

		// start inbound connectors
		for (Entry entry : inboundConnectors.entrySet()) {
			RemoteBridge remoteBridge = entry.getKey();
			try {
				entry.getValue().start();
			} catch (Exception e) {
				String msg = "Error starting inbound connector for " + remoteBridge + " : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}

		// start outbound connectors
		for (Entry entry : outboundConnectors.entrySet()) {
			RemoteBridge remoteBridge = entry.getKey();
			try {
				entry.getValue().start();
			} catch (Exception e) {
				String msg = "Error starting outbound connector for " + remoteBridge + " : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}

		LOG.info("Started");
	}

	@Override
	protected void doStop() {
		if (defaultInboundConnector != null) {
			// stop the shared inbound connector
			try {
				defaultInboundConnector.stop();
			} catch (Exception e) {
				String msg = "Error stopping inbound connector : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}
		
		// stop inbound connectors
		for (Entry entry : inboundConnectors.entrySet()) {
			RemoteBridge remoteBridge = entry.getKey();
			try {
				entry.getValue().stop();
			} catch (Exception e) {
				String msg = "Error stopping inbound connector for " + remoteBridge + " : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}

		// stop outbound connectors
		for (Entry entry : outboundConnectors.entrySet()) {
			RemoteBridge remoteBridge = entry.getKey();
			try {
				entry.getValue().stop();
			} catch (Exception e) {
				String msg = "Error stopping outbound connector for " + remoteBridge + " : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}

		LOG.info("Stopped");
	}

	@Override
	protected void doDestroy() throws Exception {
		if (defaultInboundConnector != null) {
			// destroy the shared inbound connector
			try {
				defaultInboundConnector.destroy();
			} catch (Exception e) {
				String msg = "Error destroying inbound connector : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
            defaultInboundConnector = null;
		}
		
		// destroy inbound connectors
		for (Entry entry : inboundConnectors.entrySet()) {
			RemoteBridge remoteBridge = entry.getKey();
			try {
				entry.getValue().destroy();
			} catch (Exception e) {
				String msg = "Error destroying inbound connector for " + remoteBridge + " : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}
        inboundConnectors.clear();

		// destroy outbound connectors
		for (Entry entry : outboundConnectors.entrySet()) {
			RemoteBridge remoteBridge = entry.getKey();
			try {
				entry.getValue().destroy();
			} catch (Exception e) {
				String msg = "Error destroying outbound connector for " + remoteBridge + " : " + e.getMessage();
				LOG.error(msg, e);
				throw new IllegalStateException(msg, e);
			}
		}
        outboundConnectors.clear();

		LOG.info("Destroyed");
	}

	@Override
	public BridgeDestinationsConfig getDestinationsConfig() throws JmsException {
		synchronized (lifecycleMonitor) {
			return outboundDestinations;
		}
	}

	@Override
	public void setDestinationsConfig(
			BridgeDestinationsConfig destinationsConfig) throws JmsException {
		synchronized (lifecycleMonitor) {
	
			if (destinationsConfig == null || destinationsConfig.getDestinations() == null) {
				throw new UncategorizedJmsException("Invalid destinations config");
			}
	
			// remember current state
			boolean wasRunning = isRunning();
	
			// destroy and recreate the connector
			try {
				destroy();
			} catch (Exception e) {
				LOG.error("Error destorying connector: " + e.getMessage(), e);
				// ignore and keep going??
			}
	
			this.outboundDestinations = destinationsConfig;
			
			try {
				afterPropertiesSet();
			} catch (Exception e) {
				throw new UncategorizedJmsException(e.getMessage(), e);
			}
			
			if (wasRunning) {
				start();
			}
			
			LOG.info("Outbound destinations set to " + destinationsConfig);
		}
	}

	@Override
	public void addDestinations(List destinations)
			throws JmsException {
		synchronized (lifecycleMonitor) {
			
			if (destinations == null || destinations.isEmpty()) {
				throw new IllegalArgumentException("Null or empty destinations");
			}
			
			boolean updated = false;
			for (SourceConnector connector : outboundConnectors.values()) {
				connector.addDestinations(destinations);
				if (!updated) {
					outboundDestinations = connector.getOutboundDestinations();
					updated = true;
				}
			}
		}
	}

	@Override
	public void removeDestinations(List destinations)
			throws JmsException {
		synchronized (lifecycleMonitor) {
			
			if (destinations == null || destinations.isEmpty()) {
				throw new IllegalArgumentException("Null or empty destinations");
			}
			
			boolean updated = false;
			for (SourceConnector connector : outboundConnectors.values()) {
				connector.removeDestinations(destinations);
				if (!updated) {
					outboundDestinations = connector.getOutboundDestinations();
					updated = true;
				}
			}
		}
	}

	public void addRemoteBridge(RemoteBridge remoteBridge) {
		synchronized (lifecycleMonitor) {

			if (remoteBridges.contains(remoteBridge)) {
				String msg = "Remote bridge " + remoteBridge + " already exists in Gateway";
				LOG.error(msg);
                throw new UncategorizedJmsException(msg);
            }

            remoteBridges.add(remoteBridge);

            // create connectors if initialized
            if (isInitialized()) {

                // check if the bridge needs an inbound connector or not
                if ((remoteBridge.getInboundDestinations() != null) &&
                    !remoteBridge.getInboundDestinations().equals(inboundDestinations)) {

                    TargetConnector inboundConnector;
                    try {
                        // create inbound connector
                        inboundConnector = createInboundConnector(remoteBridge);
                    } catch (Exception e) {
                        String msg = "Error creating inbound connector for " + remoteBridge + " : " + e.getMessage();
                        LOG.error(msg, e);
                        throw new UncategorizedJmsException(msg , e);
                    }

                    inboundConnectors.put(remoteBridge, inboundConnector);

                    // start connector if running
                    if (isRunning()) {
                        try {
                            inboundConnector.start();
                        } catch (Exception e) {
                            String msg = "Error starting inbound connector for " + remoteBridge + " : " + e.getMessage();
                            LOG.error(msg, e);
                            throw new UncategorizedJmsException(msg , e);
                        }
                    }
                } else {
                    LOG.warn("No custom inbound destinations in Bridge, " +
                        "no inbound connector will be created for " + remoteBridge);
                }

                // check if the bridge needs an outbound connector or not
                if ((remoteBridge.getOutboundDestinations() != null) ||
                    (this.outboundDestinations != null)) {

                    SourceConnector outboundConnector;
                    try {
                        // create outbound connector
                        outboundConnector = createOutboundConnector(remoteBridge);
                    } catch (Exception e) {
                        String msg = "Error creating outbound connector for " + remoteBridge + " : " + e.getMessage();
                        LOG.error(msg, e);
                        throw new UncategorizedJmsException(msg , e);
                    }

                    outboundConnectors.put(remoteBridge, outboundConnector);

                    // start connector if running
                    if (isRunning()) {
                        try {
                            outboundConnector.start();
                        } catch (Exception e) {
                            String msg = "Error starting outbound connector for " + remoteBridge + " : " + e.getMessage();
                            LOG.error(msg, e);
                            throw new UncategorizedJmsException(msg , e);
                        }
                    }
                } else {
                    LOG.warn("No outbound destinations in Bridge or Gateway, " +
                        "connection is unidirectional so no connector will be created for " + remoteBridge);
                }
            }
		}
	}
	
	public void removeRemoteBridge(RemoteBridge remoteBridge) {
		synchronized (lifecycleMonitor) {
            if (!remoteBridges.remove(remoteBridge)) {
                throw new UncategorizedJmsException("Not found remote bridge " + remoteBridge);
            }

            // remove and destroy connectors if initialized
            if (isInitialized()) {
                TargetConnector inboundConnector = inboundConnectors.remove(remoteBridge);
                if (inboundConnector != null) {
                    try {
                        inboundConnector.destroy();
                    } catch (Exception e) {
                        String msg = "Error removing inbound connector for " + remoteBridge + " : " + e.getMessage();
                        LOG.error(msg, e);
                        throw new UncategorizedJmsException(msg , e);
                    }
                }

                SourceConnector outboundConnector = outboundConnectors.remove(remoteBridge);
                if (outboundConnector != null) {
                    try {
                        outboundConnector.destroy();
                    } catch (Exception e) {
                        String msg = "Error removing outbound connector for " + remoteBridge + " : " + e.getMessage();
                        LOG.error(msg, e);
                        throw new UncategorizedJmsException(msg , e);
                    }
                }
            }
		}
	}
	
	public final BrokerConfig getLocalBrokerConfig() {
		return localBrokerConfig;
	}

	public final void setLocalBrokerConfig(BrokerConfig localBrokerConfig) {
		this.localBrokerConfig = localBrokerConfig;
	}

	public final BridgeDestinationsConfig getInboundDestinations() {
		return inboundDestinations;
	}

	public final void setInboundDestinations(
			BridgeDestinationsConfig inboundDestinations) {
		this.inboundDestinations = inboundDestinations;
	}

	public final BridgeDestinationsConfig getOutboundDestinations() {
		return outboundDestinations;
	}

	public final void setOutboundDestinations(
			BridgeDestinationsConfig outboundDestinations) {
		this.outboundDestinations = outboundDestinations;
	}

	public final List getRemoteBridges() {
		return remoteBridges;
	}

	public final void setRemoteBridges(List remoteBridges) {
		this.remoteBridges = remoteBridges;
	}

    public String getInboundDestinationsRef() {
        return inboundDestinationsRef;
    }

    public void setInboundDestinationsRef(String inboundDestinationsRef) {
        this.inboundDestinationsRef = inboundDestinationsRef;
    }

    public String getOutboundDestinationsRef() {
        return outboundDestinationsRef;
    }

    public void setOutboundDestinationsRef(String outboundDestinationsRef) {
        this.outboundDestinationsRef = outboundDestinationsRef;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy