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

org.modeshape.jcr.bus.ClusteredRepositoryChangeBus Maven / Gradle / Ivy

/*
 * ModeShape (http://www.modeshape.org)
 * See the COPYRIGHT.txt file distributed with this work for information
 * regarding copyright ownership.  Some portions may be licensed
 * to Red Hat, Inc. under one or more contributor license agreements.
 * See the AUTHORS.txt file in the distribution for a full listing of
 * individual contributors.
 *
 * ModeShape is free software. Unless otherwise indicated, all code in ModeShape
 * is licensed to you 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.
 *
 * ModeShape 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.modeshape.jcr.bus;

import java.util.concurrent.atomic.AtomicBoolean;
import org.jgroups.Address;
import org.jgroups.Channel;
import org.jgroups.ChannelListener;
import org.jgroups.Message;
import org.jgroups.ReceiverAdapter;
import org.jgroups.View;
import org.jgroups.util.Util;
import org.modeshape.common.SystemFailureException;
import org.modeshape.common.annotation.ThreadSafe;
import org.modeshape.common.util.CheckArg;
import org.modeshape.common.logging.Logger;
import org.modeshape.jcr.RepositoryConfiguration;
import org.modeshape.jcr.cache.change.ChangeSet;
import org.modeshape.jcr.cache.change.ChangeSetListener;
import org.modeshape.jcr.clustering.ChannelProvider;

/**
 * Implementation of a {@link ChangeBus} which can run in a cluster, via JGroups. This bus wraps around another bus, to which it
 * delegates all "local" processing of events.
 * 
 * @author Horia Chiorean
 */
@ThreadSafe
public final class ClusteredRepositoryChangeBus implements ChangeBus {

    protected static final Logger LOGGER = Logger.getLogger(ClusteredRepositoryChangeBus.class);

    /**
     * The wrapped standalone bus to which standard bus operations are delegated
     */
    protected final ChangeBus delegate;

    /**
     * The listener for channel changes.
     */
    private final Listener listener = new Listener();

    /**
     * The component that will receive the JGroups messages and broadcast them to this bus' observers.
     */
    private final Receiver receiver = new Receiver();

    /**
     * Flag that dictates whether this bus has connected to the cluster.
     */
    protected final AtomicBoolean isOpen = new AtomicBoolean(false);

    /**
     * Flag that dictates whether there are multiple participants in the cluster; if not, then the changes are propagated only to
     * the local observers.
     */
    protected final AtomicBoolean multipleAddressesInCluster = new AtomicBoolean(false);

    /**
     * The clustering configuration
     */
    protected final RepositoryConfiguration.Clustering clusteringConfiguration;

    /**
     * The JGroups channel to which all {@link #notify(ChangeSet) change notifications} will be sent and from which all changes
     * will be received and sent to the observers.
     * 

* It is important that the order of the {@link ChangeSet} instances are maintained across the cluster, and JGroups will do * this for us as long as we push all local changes into the channel and receive all local/remote changes from the channel. *

*/ private Channel channel; public ClusteredRepositoryChangeBus( RepositoryConfiguration.Clustering clusteringConfiguration, ChangeBus delegate ) { CheckArg.isNotNull(clusteringConfiguration, "clusteringConfiguration"); CheckArg.isNotNull(delegate, "delegate"); this.clusteringConfiguration = clusteringConfiguration; assert clusteringConfiguration.isEnabled(); this.delegate = delegate; } @Override public synchronized void start() throws Exception { String clusterName = clusteringConfiguration.getClusterName(); if (clusterName == null) { throw new IllegalStateException(BusI18n.clusterNameRequired.text()); } if (channel != null) { // Disconnect from any previous channel ... channel.removeChannelListener(listener); channel.setReceiver(null); } // Create the new channel by calling the delegate method ... channel = newChannel(); // Add a listener through which we'll know what's going on within the cluster ... channel.addChannelListener(listener); // Set the receiver through which we'll receive all of the changes ... channel.setReceiver(receiver); // Now connect to the cluster ... channel.connect(clusterName); // start the delegate delegate.start(); } private Channel newChannel() throws Exception { // Try to get the channel directly from the configuration (and its environment) ... Channel channel = clusteringConfiguration.getChannel(); if (channel != null) { return channel; } String lookupClassName = clusteringConfiguration.getChannelProviderClassName(); assert lookupClassName != null; Class lookupClass = Class.forName(lookupClassName); if (!ChannelProvider.class.isAssignableFrom(lookupClass)) { throw new IllegalArgumentException( "Invalid channel lookup class configured. Expected a subclass of org.modeshape.jcr.clustering.ChannelProvider. Actual class:" + lookupClass); } return ((ChannelProvider)lookupClass.newInstance()).getChannel(clusteringConfiguration); } @Override public boolean hasObservers() { return delegate.hasObservers(); } /** * Return whether this bus has been {@link #start() started} and not yet {@link #shutdown() shut down}. * * @return true if {@link #start()} has been called but {@link #shutdown()} has not, or false otherwise */ public boolean isStarted() { return channel != null; } @Override public synchronized void shutdown() { if (channel != null) { // Mark this as not accepting any more ... isOpen.set(false); try { // Disconnect from the channel and close it ... channel.removeChannelListener(listener); channel.setReceiver(null); channel.close(); } finally { channel = null; // Now that we're not receiving any more messages, shut down the delegate delegate.shutdown(); } } } @Override public void notify( ChangeSet changeSet ) { if (changeSet == null) { return; // do nothing } if (!isOpen.get()) { // The channel is not open ... return; } if (!multipleAddressesInCluster.get()) { // We are in clustered mode, but there is only one participant in the cluster (us). // So short-circuit the cluster and just notify the local observers ... if (hasObservers()) { delegate.notify(changeSet); logReceivedOperation(changeSet); } return; } // There are multiple participants in the cluster, so send all changes out to JGroups, // letting JGroups do the ordering of messages... try { logSendOperation(changeSet); byte[] data = serialize(changeSet); Message message = new Message(null, null, data); channel.send(message); } catch (IllegalStateException e) { LOGGER.warn(BusI18n.unableToNotifyChanges, clusteringConfiguration.getClusterName(), changeSet.size(), changeSet.getWorkspaceName(), changeSet.getUserId(), changeSet.getProcessKey(), changeSet.getTimestamp()); } catch (Exception e) { // Something went wrong here (this should not happen) ... String msg = BusI18n.errorSerializingChanges.text(clusteringConfiguration.getClusterName(), changeSet.size(), changeSet.getWorkspaceName(), changeSet.getUserId(), changeSet.getProcessKey(), changeSet.getTimestamp(), changeSet); throw new SystemFailureException(msg, e); } } protected final void logSendOperation( ChangeSet changeSet ) { if (LOGGER.isTraceEnabled()) { LOGGER.trace("Sending to cluster '{0}' {1} changes on workspace {2} made by {3} from process '{4}' at {5}", clusteringConfiguration.getClusterName(), changeSet.size(), changeSet.getWorkspaceName(), changeSet.getUserData(), changeSet.getProcessKey(), changeSet.getTimestamp()); } } protected final void logReceivedOperation( ChangeSet changeSet ) { if (LOGGER.isTraceEnabled()) { LOGGER.trace("Received on cluster '{0}' {1} changes on workspace {2} made by {3} from process '{4}' at {5}", clusteringConfiguration.getClusterName(), changeSet.size(), changeSet.getWorkspaceName(), changeSet.getUserId(), changeSet.getProcessKey(), changeSet.getTimestamp()); } } @Override public boolean register( ChangeSetListener observer ) { return delegate.register(observer); } @Override public boolean unregister( ChangeSetListener observer ) { return delegate.unregister(observer); } protected static byte[] serialize( ChangeSet changes ) throws Exception { return Util.objectToByteBuffer(changes); } protected static ChangeSet deserialize( byte[] data ) throws Exception { return (ChangeSet)Util.objectFromByteBuffer(data); } protected final class Receiver extends ReceiverAdapter { @Override public void block() { isOpen.set(false); } @Override public void receive( Message message ) { if (!hasObservers()) { return; } // We have at least one observer ... try { // Deserialize the changes ... ChangeSet changes = deserialize(message.getBuffer()); // and broadcast them delegate.notify(changes); logReceivedOperation(changes); } catch (Exception e) { // Something went wrong here (this should not happen) ... String msg = BusI18n.errorDeserializingChanges.text(clusteringConfiguration.getClusterName()); throw new SystemFailureException(msg, e); } } @Override public void suspect( Address suspectedMbr ) { LOGGER.error(BusI18n.memberOfClusterIsSuspect, clusteringConfiguration.getClusterName(), suspectedMbr); } @Override public void viewAccepted( View newView ) { LOGGER.trace("Members of '{0}' cluster have changed: {1}", clusteringConfiguration.getClusterName(), newView); if (newView.getMembers().size() > 1) { if (multipleAddressesInCluster.compareAndSet(false, true)) { LOGGER.debug("There are now multiple members of cluster '{0}'; changes will be propagated throughout the cluster", clusteringConfiguration.getClusterName()); } } else { if (multipleAddressesInCluster.compareAndSet(true, false)) { LOGGER.debug("There is only one member of cluster '{0}'; changes will be propagated locally only", clusteringConfiguration.getClusterName()); } } } } protected final class Listener implements ChannelListener { @Override public void channelClosed( Channel channel ) { isOpen.set(false); } @Override public void channelConnected( Channel channel ) { isOpen.set(true); } @Override public void channelDisconnected( Channel channel ) { isOpen.set(false); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy