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

bboss.org.jgroups.protocols.pbcast.Merger Maven / Gradle / Ivy

The newest version!
package bboss.org.jgroups.protocols.pbcast;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Vector;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

import bboss.org.jgroups.Address;
import bboss.org.jgroups.Event;
import bboss.org.jgroups.Membership;
import bboss.org.jgroups.MergeView;
import bboss.org.jgroups.Message;
import bboss.org.jgroups.TimeoutException;
import bboss.org.jgroups.View;
import bboss.org.jgroups.ViewId;
import bboss.org.jgroups.annotations.GuardedBy;
import bboss.org.jgroups.logging.Log;
import bboss.org.jgroups.util.Digest;
import bboss.org.jgroups.util.MergeId;
import bboss.org.jgroups.util.MutableDigest;
import bboss.org.jgroups.util.ResponseCollector;
import bboss.org.jgroups.util.Util;

/**
 * Handles merging. Called by CoordGmsImpl and ParticipantGmsImpl
 * @author Bela Ban
 * @version $Id: Merger.java,v 1.6 2010/06/11 05:41:32 belaban Exp $
 */
public class Merger {
    private final GMS                          gms;
    private final Log                          log;

    private final MergeTask                    merge_task=new MergeTask();

    /** For MERGE_REQ/MERGE_RSP correlation, contains MergeData elements */
    private final ResponseCollector merge_rsps=new ResponseCollector();

    /** For GET_DIGEST / DIGEST_RSP correlation */
    private final ResponseCollector    digest_collector=new ResponseCollector();

    /** To serialize access to merge_id */
    private final Lock                         merge_lock=new ReentrantLock();

    @GuardedBy("merge_lock")
    private MergeId                            merge_id=null;

    @GuardedBy("merge_canceller_lock")
    private Future                          merge_canceller_future=null;

    private final Lock                         merge_canceller_lock=new ReentrantLock();


    public Merger(GMS gms, Log log) {
        this.gms=gms;
        this.log=log;
    }


    /**
     * Invoked upon receiving a MERGE event from the MERGE layer. Starts the merge protocol.
     * See description of protocol in DESIGN.
     * @param views A List of different views detected by the merge protocol
     */
    public void merge(Map views) {
        if(isMergeInProgress()) {
            if(log.isTraceEnabled()) log.trace(gms.local_addr + ": merge is already running (merge_id=" + merge_id + ")");
            return;
        }

        // we need the merge *coordinators* not merge participants because not everyone can lead a merge !
        Collection
coords=Util.determineMergeCoords(views); Collection
merge_participants=Util.determineMergeParticipants(views); Membership tmp=new Membership(coords); // establish a deterministic order, so that coords can elect leader tmp.sort(); Address merge_leader=tmp.elementAt(0); if(log.isDebugEnabled()) log.debug("determining merge leader from " + merge_participants); if(merge_leader.equals(gms.local_addr)) { if(log.isDebugEnabled()) log.debug("I (" + gms.local_addr + ") will be the leader. Starting the merge task for " + merge_participants); merge_task.start(views); } else { if(log.isDebugEnabled()) log.debug("I (" + gms.local_addr + ") am not the merge leader, " + "waiting for merge leader (" + merge_leader + ") to initiate merge"); } } /** * Get the view and digest and send back both (MergeData) in the form of a MERGE_RSP to the sender. * If a merge is already in progress, send back a MergeData with the merge_rejected field set to true. * @param sender The address of the merge leader * @param merge_id The merge ID * @param mbrs The set of members from which we expect responses */ public void handleMergeRequest(Address sender, MergeId merge_id, Collection mbrs) { boolean success=matchMergeId(merge_id) || setMergeId(null, merge_id); if(!success) { if(log.isWarnEnabled()) log.warn(gms.local_addr + ": merge is already in progress"); sendMergeRejectedResponse(sender, merge_id); return; } /* Clears the view handler queue and discards all JOIN/LEAVE/MERGE requests until after the MERGE */ gms.getViewHandler().suspend(merge_id); if(log.isDebugEnabled()) log.debug(gms.local_addr + ": got merge request from " + sender + ", merge_id=" + merge_id + ", mbrs=" + mbrs); // merge the membership of the current view with mbrs List
members=new LinkedList
(); if(mbrs != null) { // didn't use a set because we didn't want to change the membership order at this time (although not incorrect) for(Address mbr: mbrs) { if(!members.contains(mbr)) members.add(mbr); } } ViewId tmp_vid=gms.view_id != null? gms.view_id.copy() : null; if(tmp_vid == null) { log.warn("view ID is null; cannot return merge response"); sendMergeRejectedResponse(sender, merge_id); return; } View view=new View(tmp_vid, new Vector
(members)); //[JGRP-524] - FLUSH and merge: flush doesn't wrap entire merge process //[JGRP-770] - Concurrent startup of many channels doesn't stabilize //[JGRP-700] - FLUSH: flushing should span merge /*if flush is in stack, let this coordinator flush its cluster island */ boolean successfulFlush=gms.startFlush(view); if(!successfulFlush) { sendMergeRejectedResponse(sender, merge_id); if(log.isWarnEnabled()) log.warn(gms.local_addr + ": flush failed; sending merge rejected message to "+ sender+ ", merge_id="+ merge_id); cancelMerge(merge_id); return; } Digest digest=fetchDigestsFromAllMembersInSubPartition(members); if(digest.size() == 0) { log.error("failed fetching digests from subpartition members; dropping merge response"); return; } sendMergeResponse(sender, view, digest, merge_id); } public void handleMergeResponse(MergeData data, MergeId merge_id) { if(!matchMergeId(merge_id)) { if(log.isErrorEnabled()) log.error(gms.local_addr + ": this.merge_id (" + this.merge_id + ") is different from merge_id (" + merge_id + ')'); return; } merge_rsps.add(data.getSender(), data); } /** * If merge_id is not equal to this.merge_id then discard. * Else cast the view/digest to all members of this group. */ public void handleMergeView(final MergeData data,final MergeId merge_id) { if(!matchMergeId(merge_id)) { if(log.isErrorEnabled()) log.error("merge_ids don't match (or are null); merge view discarded"); return; } // only send to our *current* members, if we have A and B being merged (we are B), then we would *not* // receive a VIEW_ACK from A because A doesn't see us in the pre-merge view yet and discards the view //[JGRP-700] - FLUSH: flushing should span merge //we have to send new view only to current members and we should not wait //for view acks from newly merged mebers List
newViewMembers=new Vector
(data.view.getMembers()); newViewMembers.removeAll(gms.members.getMembers()); gms.castViewChangeWithDest(data.view, data.digest, null, newViewMembers); // if we have flush in stack send ack back to merge coordinator if(gms.flushProtocolInStack) { Message ack=new Message(data.getSender(), null, null); ack.setFlag(Message.OOB); GMS.GmsHeader ack_hdr=new GMS.GmsHeader(GMS.GmsHeader.INSTALL_MERGE_VIEW_OK); ack.putHeader(gms.getId(), ack_hdr); gms.getDownProtocol().down(new Event(Event.MSG, ack)); } cancelMerge(merge_id); } public void handleMergeCancelled(MergeId merge_id) { try { gms.stopFlush(); } catch(Throwable t) { log.error("stop flush failed", t); } if(matchMergeId(merge_id)) { if(log.isDebugEnabled()) log.debug(gms.local_addr + ": merge " + merge_id + " is cancelled"); cancelMerge(merge_id); } } public void handleDigestResponse(Address sender, Digest digest) { digest_collector.add(sender, digest); } /** Send back a response containing view and digest to sender */ private void sendMergeResponse(Address sender, View view, Digest digest, MergeId merge_id) { Message msg=new Message(sender, null, null); msg.setFlag(Message.OOB); GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.MERGE_RSP); hdr.merge_id=merge_id; hdr.view=view; hdr.my_digest=digest; msg.putHeader(gms.getId(), hdr); if(log.isDebugEnabled()) log.debug(gms.local_addr + ": sending merge response=" + hdr); gms.getDownProtocol().down(new Event(Event.MSG, msg)); } /** * Sends the new view and digest to all subgroup coordinors in coords. Each coord will in turn *
    *
  1. broadcast the new view and digest to all the members of its subgroup (MergeView) *
  2. on reception of the view, if it is a MergeView, each member will set the digest and install the new view *
*/ private void sendMergeView(Collection
coords, MergeData combined_merge_data, MergeId merge_id) { if(coords == null || combined_merge_data == null) return; View view=combined_merge_data.view; Digest digest=combined_merge_data.digest; if(view == null || digest == null) { if(log.isErrorEnabled()) log.error("view or digest is null, cannot send consolidated merge view/digest"); return; } if(log.isDebugEnabled()) log.debug(gms.local_addr + ": sending merge view " + view.getVid() + " to coordinators " + coords); gms.merge_ack_collector.reset(coords); int size=gms.merge_ack_collector.size(); long timeout=gms.view_ack_collection_timeout; long start=System.currentTimeMillis(); for(Address coord:coords) { Message msg=new Message(coord, null, null); GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.INSTALL_MERGE_VIEW); hdr.view=view; hdr.my_digest=digest; hdr.merge_id=merge_id; msg.putHeader(gms.getId(), hdr); gms.getDownProtocol().down(new Event(Event.MSG, msg)); } //[JGRP-700] - FLUSH: flushing should span merge // if flush is in stack wait for acks from separated island coordinators if(gms.flushProtocolInStack) { try { gms.merge_ack_collector.waitForAllAcks(timeout); long stop=System.currentTimeMillis(); if(log.isTraceEnabled()) log.trace("received all ACKs (" + size + ") for merge view " + view + " in " + (stop - start) + "ms"); } catch(TimeoutException e) { log.warn(gms.local_addr + ": failed to collect all ACKs for merge (" + size + ") for view " + view + " after " + timeout + "ms, missing ACKs from " + gms.merge_ack_collector.printMissing()); } } } protected void sendMergeRejectedResponse(Address sender, MergeId merge_id) { Message msg=new Message(sender, null, null); msg.setFlag(Message.OOB); GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.MERGE_RSP); hdr.merge_rejected=true; hdr.merge_id=merge_id; msg.putHeader(gms.getId(), hdr); if(log.isDebugEnabled()) log.debug("merge response=" + hdr); gms.getDownProtocol().down(new Event(Event.MSG, msg)); } private void sendMergeCancelledMessage(Collection
coords, MergeId merge_id) { if(coords == null || merge_id == null) return; for(Address coord:coords) { Message msg=new Message(coord, null, null); // msg.setFlag(Message.OOB); GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.CANCEL_MERGE); hdr.merge_id=merge_id; msg.putHeader(gms.getId(), hdr); if(log.isDebugEnabled()) log.debug(gms.local_addr + ": sending cancel merge to " + coord); gms.getDownProtocol().down(new Event(Event.MSG, msg)); } } /** * Multicasts a GET_DIGEST_REQ to all current members and waits for all responses (GET_DIGEST_RSP) or N ms. * @return */ private Digest fetchDigestsFromAllMembersInSubPartition(List
current_mbrs) { if(current_mbrs == null) return null; GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.GET_DIGEST_REQ); Message get_digest_req=new Message(); get_digest_req.setFlag(Message.OOB); get_digest_req.putHeader(gms.getId(), hdr); long max_wait_time=gms.merge_timeout > 0? gms.merge_timeout / 2 : 2000L; digest_collector.reset(current_mbrs); // add my own digest first Digest digest=(Digest)gms.getDownProtocol().down(Event.GET_DIGEST_EVT); digest_collector.add(gms.local_addr, digest); gms.getDownProtocol().down(new Event(Event.MSG, get_digest_req)); digest_collector.waitForAllResponses(max_wait_time); if(log.isDebugEnabled()) { if(digest_collector.hasAllResponses()) log.debug(gms.local_addr + ": fetched all digests for " + current_mbrs); else log.debug(gms.local_addr + ": fetched incomplete digests (after timeout of " + max_wait_time + ") ms for " + current_mbrs); } Map responses=new HashMap(digest_collector.getResults()); MutableDigest retval=new MutableDigest(responses.size()); for(Digest dig: responses.values()) { if(dig != null) retval.add(dig); } return retval; } /** * Fetches the digests from all members and installs them again. Used only for diagnosis and support; don't * use this otherwise ! */ void fixDigests() { Digest digest=fetchDigestsFromAllMembersInSubPartition(gms.view.getMembers()); Message msg=new Message(); GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.INSTALL_DIGEST); hdr.my_digest=digest; msg.putHeader(gms.getId(), hdr); gms.getDownProtocol().down(new Event(Event.MSG, msg)); } void stop() { merge_task.stop(); } void cancelMerge(MergeId id) { if(setMergeId(id, null)) { merge_task.stop(); merge_rsps.reset(); gms.getViewHandler().resume(id); } } private boolean setMergeId(MergeId expected, MergeId new_value) { merge_lock.lock(); try { boolean match=Util.match(this.merge_id, expected); if(match) { this.merge_id=new_value; stopMergeCanceller(); if(this.merge_id != null) startMergeCanceller(); } return match; } finally { merge_lock.unlock(); } } /** Only used for testing, might get removed any time. Do not use ! */ public MergeId getMergeId() { merge_lock.lock(); try { return merge_id; } finally { merge_lock.unlock(); } } private boolean isMergeInProgress() { merge_lock.lock(); try { return merge_id != null; } finally { merge_lock.unlock(); } } private boolean matchMergeId(MergeId id) { merge_lock.lock(); try { return Util.match(this.merge_id, id); } finally { merge_lock.unlock(); } } private void startMergeCanceller() { merge_canceller_lock.lock(); try { if(merge_canceller_future == null || merge_canceller_future.isDone()) { MergeCanceller task=new MergeCanceller(this.merge_id); merge_canceller_future=gms.timer.schedule(task, (long)(gms.merge_timeout * 1.5), TimeUnit.MILLISECONDS); } } finally { merge_canceller_lock.unlock(); } } private void stopMergeCanceller() { merge_canceller_lock.lock(); try { if(merge_canceller_future != null) { merge_canceller_future.cancel(true); merge_canceller_future=null; } } finally { merge_canceller_lock.unlock(); } } /** * Starts the merge protocol (only run by the merge leader). Essentially sends a MERGE_REQ to all * coordinators of all subgroups found. Each coord receives its digest and view and returns it. * The leader then computes the digest and view for the new group from the return values. Finally, it * sends this merged view/digest to all subgroup coordinators; each coordinator will install it in their * subgroup. */ class MergeTask implements Runnable { private Thread thread=null; /** List of all subpartition coordinators and their members */ private final ConcurrentMap> coords=new ConcurrentHashMap>(); /** * @param views Guaranteed to be non-null and to have >= 2 members, or else this thread would not be started */ public synchronized void start(Map views) { if(thread == null || thread.isAlive()) { this.coords.clear(); // Add all different coordinators of the views into the hashmap and sets their members: Collection
coordinators=Util.determineMergeCoords(views); for(Address coord: coordinators) { View view=views.get(coord); if(view != null) this.coords.put(coord, new ArrayList
(view.getMembers())); } // For the merge participants which are not coordinator, we simply add them, and the associated // membership list consists only of themselves Collection
merge_participants=Util.determineMergeParticipants(views); merge_participants.removeAll(coordinators); for(Address merge_participant: merge_participants) coords.putIfAbsent(merge_participant, Arrays.asList(merge_participant)); thread=gms.getThreadFactory().newThread(this, "MergeTask"); thread.setDaemon(true); thread.start(); } } public synchronized void stop() { Thread tmp=thread; if(thread != null && thread.isAlive()) tmp.interrupt(); thread=null; } /** Runs the merge protocol as a leader */ public void run() { // 1. Generate merge_id MergeId new_merge_id=MergeId.create(gms.local_addr); Collection
coordsCopy=null; try { boolean success=setMergeId(null, new_merge_id); if(!success) { log.warn("failed to set my own merge_id (" + merge_id + ") to " + new_merge_id); return; } coordsCopy=new ArrayList
(coords.keySet()); /* 2. Fetch the current Views/Digests from all subgroup coordinators */ success=getMergeDataFromSubgroupCoordinators(coords, new_merge_id, gms.merge_timeout); if(!success) throw new Exception("merge leader did not get data from all partition coordinators " + coords.keySet()); /* 3. Remove rejected MergeData elements from merge_rsp and coords (so we'll send the new view * only to members who accepted the merge request) */ removeRejectedMergeRequests(coords.keySet()); if(merge_rsps.size() == 0) throw new Exception("did not get any merge responses from partition coordinators"); if(!coords.keySet().contains(gms.local_addr)) // another member might have invoked a merge req on us before we got there... throw new Exception("merge leader rejected merge request"); /* 4. Combine all views and digests into 1 View/1 Digest */ Vector merge_data=new Vector(merge_rsps.getResults().values()); MergeData combined_merge_data=consolidateMergeData(merge_data); if(combined_merge_data == null) throw new Exception("could not consolidate merge"); /* 4. Send the new View/Digest to all coordinators (including myself). On reception, they will install the digest and view in all of their subgroup members */ sendMergeView(coords.keySet(), combined_merge_data, new_merge_id); } catch(Throwable ex) { if(log.isWarnEnabled()) log.warn(gms.local_addr + ": " + ex.getLocalizedMessage() + ", merge is cancelled"); sendMergeCancelledMessage(coordsCopy, new_merge_id); } finally { gms.getViewHandler().resume(new_merge_id); stopMergeCanceller(); // this is probably not necessary /*5. if flush is in stack stop the flush for entire cluster [JGRP-700] - FLUSH: flushing should span merge */ gms.stopFlush(); if(log.isDebugEnabled()) log.debug(gms.local_addr + ": merge leader completed merge task"); thread=null; } } /** * Sends a MERGE_REQ to all coords and populates a list of MergeData (in merge_rsps). Returns after coords.size() * response have been received, or timeout msecs have elapsed (whichever is first).

* If a subgroup coordinator rejects the MERGE_REQ (e.g. because of participation in a different merge), * that member will be removed from coords ! * @param coords A map of coordinatgor addresses and associated membership lists * @param new_merge_id The new merge id * @param timeout Max number of msecs to wait for the merge responses from the subgroup coords */ private boolean getMergeDataFromSubgroupCoordinators(Map> coords, MergeId new_merge_id, long timeout) { boolean gotAllResponses; long start=System.currentTimeMillis(); merge_rsps.reset(coords.keySet()); if(log.isDebugEnabled()) log.debug(gms.local_addr + ": sending MERGE_REQ to " + coords.keySet()); for(Map.Entry> entry: coords.entrySet()) { Address coord=entry.getKey(); Collection

mbrs=entry.getValue(); Message msg=new Message(coord, null, null); msg.setFlag(Message.OOB); GMS.GmsHeader hdr=new GMS.GmsHeader(GMS.GmsHeader.MERGE_REQ, mbrs); hdr.mbr=gms.local_addr; hdr.merge_id=new_merge_id; msg.putHeader(gms.getId(), hdr); gms.getDownProtocol().down(new Event(Event.MSG, msg)); } // wait until num_rsps_expected >= num_rsps or timeout elapsed merge_rsps.waitForAllResponses(timeout); gotAllResponses=merge_rsps.hasAllResponses(); long stop=System.currentTimeMillis(); if(log.isDebugEnabled()) log.debug(gms.local_addr + ": collected " + merge_rsps.size() + " merge response(s) in " + (stop-start) + " ms"); return gotAllResponses; } /** Removed rejected merge requests from merge_rsps and coords. This method has a lock on merge_rsps */ private void removeRejectedMergeRequests(Collection
coords) { for(Map.Entry entry: merge_rsps.getResults().entrySet()) { Address member=entry.getKey(); MergeData data=entry.getValue(); if(data.merge_rejected) { if(data.getSender() != null) coords.remove(data.getSender()); merge_rsps.remove(member); } } } /** * Merge all MergeData. All MergeData elements should be disjunct (both views and digests). However, * this method is prepared to resolve duplicate entries (for the same member). Resolution strategy for * views is to merge only 1 of the duplicate members. Resolution strategy for digests is to take the higher * seqnos for duplicate digests.

* After merging all members into a Membership and subsequent sorting, the first member of the sorted membership * will be the new coordinator. This method has a lock on merge_rsps. * @param merge_rsps A list of MergeData items. Elements with merge_rejected=true were removed before. Is guaranteed * not to be null and to contain at least 1 member. */ private MergeData consolidateMergeData(Vector merge_rsps) { long logical_time=0; // for new_vid Membership new_mbrs=new Membership(); Vector subgroups=new Vector(11); // contains a list of Views, each View is a subgroup for(MergeData tmp_data: merge_rsps) { View tmp_view=tmp_data.getView(); if(tmp_view != null) { ViewId tmp_vid=tmp_view.getVid(); if(tmp_vid != null) { // compute the new view id (max of all vids +1) logical_time=Math.max(logical_time, tmp_vid.getId()); } // merge all membership lists into one (prevent duplicates) new_mbrs.add(tmp_view.getMembers()); subgroups.addElement((View)tmp_view.clone()); } } // the new coordinator is the first member of the consolidated & sorted membership list new_mbrs.sort(); Address new_coord = new_mbrs.size() > 0 ? new_mbrs.elementAt(0) : null; if(new_coord == null) { if(log.isErrorEnabled()) log.error("new_coord == null"); return null; } // should be the highest view ID seen up to now plus 1 ViewId new_vid=new ViewId(new_coord, logical_time + 1); // determine the new view MergeView new_view=new MergeView(new_vid, new_mbrs.getMembers(), subgroups); // determine the new digest Digest new_digest=consolidateDigests(merge_rsps, new_mbrs.size()); if(new_digest == null) { if(log.isErrorEnabled()) log.error("Merge leader " + gms.local_addr + ": could not consolidate digest for merge"); return null; } if(log.isDebugEnabled()) log.debug("Merge leader " + gms.local_addr + ": consolidated view=" + new_view + "\nconsolidated digest=" + new_digest); return new MergeData(gms.local_addr, new_view, new_digest); } /** * Merge all digests into one. For each sender, the new value is min(low_seqno), max(high_seqno), * max(high_seqno_seen). This method has a lock on merge_rsps */ private Digest consolidateDigests(Vector merge_rsps, int num_mbrs) { MutableDigest retval=new MutableDigest(num_mbrs); for(MergeData data:merge_rsps) { Digest tmp_digest=data.getDigest(); if(tmp_digest == null) { if(log.isErrorEnabled()) log.error("tmp_digest == null; skipping"); continue; } retval.merge(tmp_digest); } return retval.copy(); } } private class MergeCanceller implements Runnable { private final MergeId my_merge_id; MergeCanceller(MergeId my_merge_id) { this.my_merge_id=my_merge_id; } public void run() { cancelMerge(my_merge_id); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy