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

org.jgroups.blocks.GroupRequest Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Beta1
Show newest version
package org.jgroups.blocks;


import org.jgroups.Address;
import org.jgroups.Message;
import org.jgroups.View;
import org.jgroups.annotations.GuardedBy;
import org.jgroups.protocols.relay.SiteAddress;
import org.jgroups.util.Rsp;
import org.jgroups.util.RspList;

import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;


/**
 * Sends a message to all members of the group and waits for all responses (or
 * timeout). Returns a boolean value (success or failure). Results (if any) can
 * be retrieved when done.
 * 

* The supported transport to send requests is currently either a * RequestCorrelator or a generic Transport. One of them has to be given in the * constructor. It will then be used to send a request. When a message is * received by either one, the receiveResponse() of this class has to be called * (this class does not actively receive requests/responses itself). Also, when * a view change or suspicion is received, the methods viewChange() or suspect() * of this class have to be called. *

* When started, an array of responses, correlating to the membership, is * created. Each response is added to the corresponding field in the array. When * all fields have been set, the algorithm terminates. This algorithm can * optionally use a suspicion service (failure detector) to detect (and exclude * from the membership) fauly members. If no suspicion service is available, * timeouts can be used instead (see execute()). When done, a * list of suspected members can be retrieved. *

* Because a channel might deliver requests, and responses to different * requests, the GroupRequest class cannot itself receive and * process requests/responses from the channel. A mechanism outside this class * has to do this; it has to determine what the responses are for the message * sent by the execute() method and call * receiveResponse() to do so. *

* Requirements: lossless delivery, e.g. acknowledgment-based message * confirmation. * * @author Bela Ban */ public class GroupRequest extends Request { /** Correlates requests and responses */ @GuardedBy("lock") private final Map> requests; @GuardedBy("lock") int num_valid; // the number of valid responses (values or exceptions that passed the response filter) @GuardedBy("lock") int num_received; // number of responses (values, exceptions or suspicions) /** * @param msg The message to be sent * @param corr The request correlator to be used. A request correlator sends requests tagged with a unique ID and * notifies the sender when matching responses are received. The reason GroupRequest uses * it instead of a Transport is that multiple requests/responses might be sent/received concurrently * @param targets The targets, which are supposed to receive the message. Any receiver not in this set will * discard the message. Targets are always a subset of the current membership * @param options The request options to be used for this call */ public GroupRequest(Message msg, RequestCorrelator corr, Collection

targets, RequestOptions options) { super(msg, corr, options); int size=targets.size(); requests=new HashMap<>(size); setTargets(targets); } public GroupRequest(Message msg, RequestCorrelator corr, Address target, RequestOptions options) { super(msg, corr, options); requests=new HashMap<>(1); setTarget(target); } public boolean getAnycasting() { return options.getAnycasting(); } public void setAnycasting(boolean anycasting) { options.setAnycasting(anycasting); } public void sendRequest() throws Exception { sendRequest(requests.keySet(), req_id); } /* ---------------------- Interface RspCollector -------------------------- */ /** * Callback (called by RequestCorrelator or Transport). * Adds a response to the response table. When all responses have been received, * execute() returns. */ @SuppressWarnings("unchecked") public void receiveResponse(Object response_value, Address sender, boolean is_exception) { if(done) return; Rsp rsp=requests.get(sender); if(rsp == null) return; RspFilter rsp_filter=options.getRspFilter(); boolean responseReceived=false; lock.lock(); try { if(!rsp.wasReceived()) { if(!(rsp.wasSuspected() || rsp.wasUnreachable())) num_received++; if((responseReceived=(rsp_filter == null) || rsp_filter.isAcceptable(response_value, sender))) { if(is_exception && response_value instanceof Throwable) rsp.setException((Throwable)response_value); else rsp.setValue((T)response_value); num_valid++; } } done=responsesComplete() || (rsp_filter != null && !rsp_filter.needMoreResponses()); if(responseReceived || done) cond.signal(true); // wakes up execute() if(done && corr != null) corr.done(req_id); } finally { lock.unlock(); } if(responseReceived || done) checkCompletion(this); } /** * Callback (called by RequestCorrelator or Transport). * Report to GroupRequest that a member is reported as faulty (suspected). * This method would probably be called when getting a suspect message from a failure detector * (where available). It is used to exclude faulty members from the response list. */ public void suspect(Address suspected_member) { if(suspected_member == null) return; boolean changed=false; Rsp rsp=requests.get(suspected_member); if(rsp != null) { if(rsp.setSuspected()) { changed=true; lock.lock(); try { if(!(rsp.wasReceived() || rsp.wasUnreachable())) num_received++; cond.signal(true); } finally { lock.unlock(); } } } if(changed) checkCompletion(this); } public void siteUnreachable(String site) { boolean changed=false; for(Map.Entry> entry: requests.entrySet()) { Address member=entry.getKey(); if(!(member instanceof SiteAddress)) continue; SiteAddress addr=(SiteAddress)member; if(addr.getSite().equals(site)) { Rsp rsp=entry.getValue(); if(rsp != null) { if(rsp.setUnreachable()) { changed=true; lock.lock(); try { if(!(rsp.wasReceived() || rsp.wasSuspected())) num_received++; cond.signal(true); } finally { lock.unlock(); } } } if(changed) checkCompletion(this); } } } /** * Any member of 'membership' that is not in the new view is flagged as * SUSPECTED. Any member in the new view that is not in the * membership (ie, the set of responses expected for the current RPC) will * not be added to it. If we did this we might run into the * following problem: *
    *
  • Membership is {A,B} *
  • A sends a synchronous group RPC (which sleeps for 60 secs in the * invocation handler) *
  • C joins while A waits for responses from A and B *
  • If this would generate a new view {A,B,C} and if this expanded the * response set to {A,B,C}, A would wait forever on C's response because C * never received the request in the first place, therefore won't send a * response. *
*/ public void viewChange(View new_view) { if(new_view == null || requests == null || requests.isEmpty()) return; List
mbrs=new_view.getMembers(); if(mbrs == null) return; boolean changed=false; lock.lock(); try { for(Map.Entry> entry: requests.entrySet()) { Address mbr=entry.getKey(); // SiteAddresses are not checked as they might be in a different cluster if(!(mbr instanceof SiteAddress) && !mbrs.contains(mbr)) { Rsp rsp=entry.getValue(); if(rsp.setSuspected()) { if(!(rsp.wasReceived() || rsp.wasUnreachable())) num_received++; changed=true; } } } if(changed) cond.signal(true); } finally { lock.unlock(); } if(changed) checkCompletion(this); } /** Marks all responses with an exception (unless a response was already marked as done) */ public void transportClosed() { boolean changed=false; lock.lock(); try { for(Map.Entry> entry: requests.entrySet()) { Rsp rsp=entry.getValue(); if(rsp != null && !(rsp.wasReceived() || rsp.wasSuspected() || rsp.wasUnreachable())) { rsp.setException(new IllegalStateException("transport was closed")); num_received++; changed=true; } } if(changed) { cond.signal(true); } } finally { lock.unlock(); } if(changed) checkCompletion(this); } /* -------------------- End of Interface RspCollector ----------------------------------- */ /** Returns the results as a RspList */ public RspList getResults() { Collection> rsps=requests.values(); return new RspList<>(rsps); } public RspList get() throws InterruptedException, ExecutionException { lock.lock(); try { waitForResults(0); } finally { lock.unlock(); } return getResults(); } public RspList get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { boolean ok; lock.lock(); try { ok=waitForResults(unit.toMillis(timeout)); } finally { lock.unlock(); } if(!ok) throw new TimeoutException(); return getResults(); } public String toString() { StringBuilder ret=new StringBuilder(128); ret.append(super.toString()); if(!requests.isEmpty()) { ret.append(", entries:\n"); for(Map.Entry> entry: requests.entrySet()) { Address mbr=entry.getKey(); Rsp rsp=entry.getValue(); ret.append(mbr).append(": ").append(rsp).append("\n"); } } return ret.toString(); } /* --------------------------------- Private Methods -------------------------------------*/ private void setTarget(Address mbr) { requests.put(mbr, new Rsp(mbr)); } private void setTargets(Collection
mbrs) { for(Address mbr: mbrs) requests.put(mbr, new Rsp(mbr)); } private static int determineMajority(int i) { return i < 2? i : (i / 2) + 1; } private void sendRequest(final Collection
targetMembers, long requestId) throws Exception { try { corr.sendRequest(requestId, targetMembers, request_msg, options.getMode() == ResponseMode.GET_NONE? null : this, options); } catch(Exception ex) { if(corr != null) corr.done(requestId); throw ex; } } @GuardedBy("lock") protected boolean responsesComplete() { if(done) return true; final int num_total=requests.size(); switch(options.getMode()) { case GET_FIRST: return num_valid >= 1 || num_received >= num_total; case GET_ALL: return num_valid >= num_total || num_received >= num_total; case GET_MAJORITY: int majority=determineMajority(num_total); return num_valid >= majority || num_received >= num_total; case GET_NONE: return true; default: if(log.isErrorEnabled()) log.error("rsp_mode " + options.getMode() + " unknown !"); break; } return false; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy