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

org.modeshape.jcr.journal.ClusteredJournal Maven / Gradle / Ivy

There is a newer version: 5.4.1.Final
Show newest version
/*
 * ModeShape (http://www.modeshape.org)
 *
 * 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.modeshape.jcr.journal;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import org.infinispan.schematic.document.ThreadSafe;
import org.joda.time.DateTime;
import org.modeshape.common.SystemFailureException;
import org.modeshape.common.logging.Logger;
import org.modeshape.common.util.CheckArg;
import org.modeshape.jcr.JcrI18n;
import org.modeshape.jcr.cache.change.ChangeSet;
import org.modeshape.jcr.clustering.ClusteringService;
import org.modeshape.jcr.clustering.MessageConsumer;

/**
 * A {@link ChangeJournal} implementation which runs in a cluster and which attempts to reconcile with other members of the cluster
 * on startup in order to retrieve missed/lost records.
 *
 * @author Horia Chiorean ([email protected])
 */
@ThreadSafe
public class ClusteredJournal extends MessageConsumer implements ChangeJournal {

    private final static Logger LOGGER  = Logger.getLogger(ClusteredJournal.class);
    private final static int MAX_MINUTES_TO_WAIT_FOR_RECONCILIATION = 5;

    private final LocalJournal localJournal;
    private final ClusteringService clusteringService;
    private final AtomicInteger expectedNumberOfDeltaResponses;

    private CountDownLatch reconciliationLatch = null;

    /**
     * Creates a new clustered journal
     *
     * @param localJournal the local {@link ChangeJournal} which will
     * @param clusteringService an {@link ClusteringService} instance.
     */
    public ClusteredJournal( LocalJournal localJournal,
                             ClusteringService clusteringService ) {
        super(DeltaMessage.class);

        CheckArg.isNotNull(localJournal, "localJournal");
        CheckArg.isNotNull(clusteringService, "clusteringService");

        this.clusteringService = clusteringService;
        this.localJournal = localJournal.withSearchTimeDelta(clusteringService.getMaxAllowedClockDelayMillis());
        this.expectedNumberOfDeltaResponses = new AtomicInteger(0);
    }

    @Override
    public void notify( ChangeSet changeSet ) {
        localJournal.notify(changeSet);
    }

    @Override
    public void start() throws Exception {
        // make sure the clustering service is open
        if (!clusteringService.isOpen())  {
            throw new IllegalStateException("The clustering service has not been started");
        }

        localJournal.start();

        //make sure this process can always process delta messages
        clusteringService.addConsumer(this);

        if (clusteringService.multipleMembersInCluster()) {
            // we expect to receive delta responses equal to how many members the cluster has
            int numberOfExpectedResponses = clusteringService.membersInCluster() - 1;
            reconciliationLatch = new CountDownLatch(numberOfExpectedResponses);
            this.expectedNumberOfDeltaResponses.compareAndSet(0, numberOfExpectedResponses);
            if (LOGGER.isDebugEnabled()) {
                LOGGER.debug("Sending delta request from journal {0} as part of cluster {1}", journalId(), clusterName());
            }
            JournalRecord lastRecord = lastRecord();
            DateTime lastChangeSetTimeMillis = lastRecord != null ? new DateTime(lastRecord.getChangeTimeMillis()) : null;
            clusteringService.sendMessage(DeltaMessage.request(journalId(), lastChangeSetTimeMillis));
            waitForReconciliationToComplete();
        } else {
            this.reconciliationLatch = new CountDownLatch(0);
        }
    }

    private void waitForReconciliationToComplete() {
        try {
            reconciliationLatch.await(MAX_MINUTES_TO_WAIT_FOR_RECONCILIATION, TimeUnit.MINUTES);
        } catch (InterruptedException e) {
            Thread.interrupted();
        }

        if (!deltaReconciliationCompleted()) {
            throw new SystemFailureException(JcrI18n.journalHasNotCompletedReconciliation.text(journalId(), clusterName(),
                                                                                               MAX_MINUTES_TO_WAIT_FOR_RECONCILIATION));
        }
    }

    @Override
    public void shutdown() {
        localJournal.shutdown();
    }

    @Override
    public void removeOldRecords() {
        localJournal.removeOldRecords();
    }

    @Override
    public Records allRecords( boolean descendingOrder ) {
        return localJournal.allRecords(descendingOrder);
    }

    @Override
    public JournalRecord lastRecord() {
        return localJournal.lastRecord();
    }

    @Override
    public Records recordsNewerThan( DateTime changeSetTime,
                                     boolean inclusive,
                                     boolean descendingOrder ) {
        return localJournal.recordsNewerThan(changeSetTime, inclusive, descendingOrder);
    }

    @Override
    public void addRecords( JournalRecord... records ) {
        localJournal.addRecords(records);
    }

    @Override
    public String journalId() {
        return localJournal.journalId();
    }

    @Override
    public void consume( DeltaMessage message ) {
        if (message instanceof DeltaMessage.DeltaRequest)  {
            processDeltaRequest((DeltaMessage.DeltaRequest)message);
            return;
        }
        if (message instanceof DeltaMessage.DeltaResponse) {
            processDeltaResponse((DeltaMessage.DeltaResponse)message);
            return;
        }
        if (message instanceof DeltaMessage.DeltaStillReconciling) {
            processStillReconciling((DeltaMessage.DeltaStillReconciling)message);
        }
    }

    private boolean deltaReconciliationCompleted() {
        return reconciliationLatch.getCount() == 0;
    }

    private void processStillReconciling( DeltaMessage.DeltaStillReconciling message ) {
        if (LOGGER.isDebugEnabled()) {
            LOGGER.debug("Journal {0} received response from journal {1}: still reconciling", journalId(), message.getJournalId());
        }
        //we've received a response, but the node from which we've received the response is still reconciling
        reconciliationLatch.countDown();
    }

    private void processDeltaRequest(DeltaMessage.DeltaRequest request) {
        if (request.getJournalId().equals(journalId())) {
            //we MUST discard own messages, because JGroups will broadcast these as well
            LOGGER.debug("Journal {0} discarding delta request from itself", journalId());
            return;
        }

        if (!deltaReconciliationCompleted()) {
            //if this clustered journal has not completed reconciling itself, it cannot send anything
            LOGGER.debug("Journal {0} is still reconciling, cannot send delta to journal {1}", journalId(), request.getJournalId());
            clusteringService.sendMessage(DeltaMessage.stillReconciling(journalId()));
            return;
        }

        Records delta = recordsNewerThan(request.getLastChangeSetTime(), false, false);
        List deltaList = new ArrayList(delta.size());
        for (JournalRecord record : delta) {
            deltaList.add(record);
        }
        if (LOGGER.isDebugEnabled()) {
            LOGGER.debug("Journal {0} sending delta response to journal {1} as part of cluster {2}. Delta size is {3}",
                         journalId(),
                         request.getJournalId(),
                         clusterName(),
                         delta.size());
        }
        clusteringService.sendMessage(DeltaMessage.response(journalId(), deltaList));
    }

    private void processDeltaResponse(DeltaMessage.DeltaResponse response) {
        List records = response.getRecords();
        if (LOGGER.isDebugEnabled()) {
            LOGGER.debug("Journal {0} received delta response from journal {1} as part of cluster {2}. Delta size: {3} records.",
                         journalId(),
                         response.getJournalId(),
                         clusterName(),
                         records.size());
        }

        if (!records.isEmpty()) {
            //make sure that a new timestamp is not generated for those records and whatever comes in the response is used.
            localJournal.addRecords(records.toArray(new JournalRecord[0]));
        }
        reconciliationLatch.countDown();
    }

    protected ClusteringService clusteringService() {
        return ClusteredJournal.this.clusteringService;
    }

    protected LocalJournal localJournal() {
        return localJournal;
    }

    protected String clusterName() {
        return clusteringService().clusterName();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy