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

org.jgroups.blocks.MessageDispatcher 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.Final
Show newest version

package org.jgroups.blocks;

import org.jgroups.*;
import org.jgroups.logging.Log;
import org.jgroups.logging.LogFactory;
import org.jgroups.protocols.TP;
import org.jgroups.protocols.relay.SiteAddress;
import org.jgroups.stack.Protocol;
import org.jgroups.stack.StateTransferInfo;
import org.jgroups.util.*;

import java.io.*;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.TimeoutException;
import java.util.stream.Stream;


/**
 * Provides synchronous and asynchronous message sending with request-response
 * correlation; i.e., matching responses with the original request.
 * It also offers push-style message reception (by internally using the PullPushAdapter).
 * 

* Channels are simple patterns to asynchronously send a receive messages. * However, a significant number of communication patterns in group communication * require synchronous communication. For example, a sender would like to send a * message to the group and wait for all responses. Or another application would * like to send a message to the group and wait only until the majority of the * receivers have sent a response, or until a timeout occurred. MessageDispatcher * offers a combination of the above pattern with other patterns. *

* Used on top of channel to implement group requests. Client's {@code handle()} * method is called when request is received. Is the equivalent of RpcProtocol on * the application instead of protocol level. * * @author Bela Ban */ public class MessageDispatcher implements RequestHandler, Closeable, ChannelListener { protected JChannel channel; protected RequestCorrelator corr; protected MembershipListener membership_listener; protected StateListener state_listener; protected RequestHandler req_handler; protected boolean async_dispatching; protected boolean wrap_exceptions; protected ProtocolAdapter prot_adapter; protected volatile Collection

members=new HashSet<>(); protected Address local_addr; protected final Log log=LogFactory.getLog(MessageDispatcher.class); protected final RpcStats rpc_stats=new RpcStats(false); protected static final RspList empty_rsplist=new RspList(); protected static final GroupRequest empty_group_request; static { empty_group_request=new GroupRequest<>(null, Collections.emptyList(), RequestOptions.SYNC()); empty_group_request.complete(empty_rsplist); } public MessageDispatcher() { } public MessageDispatcher(JChannel channel) { this.channel=channel; prot_adapter=new ProtocolAdapter(); if(channel != null) { channel.addChannelListener(this); local_addr=channel.getAddress(); installUpHandler(prot_adapter, true); } start(); } public MessageDispatcher(JChannel channel, RequestHandler req_handler) { this(channel); setRequestHandler(req_handler); } public JChannel getChannel() {return channel;} public RequestCorrelator getCorrelator() {return corr;} public RequestCorrelator correlator() {return corr;} public boolean getAsyncDispatching() {return async_dispatching;} public boolean asyncDispatching() {return async_dispatching;} public boolean getWrapExceptions() {return wrap_exceptions;} public boolean wrapExceptions() {return wrap_exceptions;} public UpHandler getProtocolAdapter() {return prot_adapter;} public UpHandler protocolAdapter() {return prot_adapter;} public RpcStats getRpcStats() {return rpc_stats;} public RpcStats rpcStats() {return rpc_stats;} public boolean getExtendedStats() {return rpc_stats.extendedStats();} public boolean extendedStats() {return rpc_stats.extendedStats();} public X setExtendedStats(boolean fl) {return extendedStats(fl);} public X extendedStats(boolean fl) {rpc_stats.extendedStats(fl); return (X)this;} public X setChannel(JChannel ch) { if(ch == null) return (X)this; this.channel=ch; if(ch != null) { local_addr=channel.getAddress(); ch.addChannelListener(this); } if(prot_adapter == null) prot_adapter=new ProtocolAdapter(); // Don't force installing the UpHandler so subclasses can use this method return installUpHandler(prot_adapter, false); } public X setCorrelator(RequestCorrelator c) {return correlator(c);} public X correlator(RequestCorrelator c) { if(c == null) return (X)this; stop(); this.corr=c; corr.asyncDispatching(this.async_dispatching).wrapExceptions(this.wrap_exceptions); start(); return (X)this; } public X setMembershipListener(MembershipListener l) { membership_listener=l; return (X)this; } public X setStateListener(StateListener sl) { this.state_listener=sl; return (X)this; } public X setRequestHandler(RequestHandler rh) { req_handler=rh; return (X)this; } public X setAsynDispatching(boolean flag) {return asyncDispatching(flag);} public X asyncDispatching(boolean flag) { async_dispatching=flag; if(corr != null) corr.asyncDispatching(flag); return (X)this; } public X setWrapExceptions(boolean flag) {return wrapExceptions(flag);} public X wrapExceptions(boolean flag) { wrap_exceptions=flag; if(corr != null) corr.wrapExceptions(flag); return (X)this; } protected X setMembers(List
new_mbrs) { if(new_mbrs != null) members=new HashSet<>(new_mbrs); // volatile write - seen by a subsequent read return (X)this; } public X start() { if(corr == null) corr=createRequestCorrelator(prot_adapter, this, local_addr) .asyncDispatching(async_dispatching).wrapExceptions(this.wrap_exceptions); correlatorStarted(); corr.start(); if(channel != null) { List
tmp_mbrs=channel.getView() != null ? channel.getView().getMembers() : null; setMembers(tmp_mbrs); if(channel instanceof JChannel) { TP transport=channel.getProtocolStack().getTransport(); corr.registerProbeHandler(transport); } } return (X)this; } protected static RequestCorrelator createRequestCorrelator(Protocol transport, RequestHandler handler, Address local_addr) { return new RequestCorrelator(transport, handler, local_addr); } protected void correlatorStarted() { ; } @Override public void close() throws IOException {stop();} public X stop() { if(corr != null) { corr.stop(); if(channel instanceof JChannel) { TP transport=channel.getProtocolStack().getTransport(); corr.unregisterProbeHandler(transport); } } return (X)this; } /** * Sets the given UpHandler as the UpHandler for the channel. If the relevant handler is already installed, * the {@code canReplace} controls whether this method replaces it (after logging a WARN) or simply * leaves {@code handler} uninstalled.

* Passing {@code false} as the {@code canReplace} value allows callers to use this method to install defaults * without concern about inadvertently overriding * * @param handler the UpHandler to install * @param canReplace {@code true} if an existing Channel upHandler can be replaced; {@code false} * if this method shouldn't install */ protected X installUpHandler(UpHandler handler, boolean canReplace) { UpHandler existing = channel.getUpHandler(); if (existing == null) channel.setUpHandler(handler); else if(canReplace) { log.warn("Channel already has an up handler installed (%s) but now it is being overridden", existing); channel.setUpHandler(handler); } return (X)this; } /** * Sends a message to all members and expects responses from members in dests (if non-null). * @param dests A list of group members from which to expect responses (if the call is blocking). * @param data The buffer * @param offset the offset into data * @param length the number of bytes to send * @param opts A set of options that govern the call. See {@link org.jgroups.blocks.RequestOptions} for details * @return RspList A list of Rsp elements, or null if the RPC is asynchronous * @throws Exception If the request cannot be sent * @since 4.0 */ public RspList castMessage(Collection

dests, byte[] data, int offset, int length, RequestOptions opts) throws Exception { return castMessage(dests, new Buffer(data, offset, length), opts); } /** * Sends a message to all members and expects responses from members in dests (if non-null). * @param dests A list of group members from which to expect responses (if the call is blocking). * @param data The message to be sent * @param opts A set of options that govern the call. See {@link org.jgroups.blocks.RequestOptions} for details * @return RspList A list of Rsp elements, or null if the RPC is asynchronous * @throws Exception If the request cannot be sent * @since 2.9 */ public RspList castMessage(final Collection
dests, Buffer data, RequestOptions opts) throws Exception { GroupRequest req=cast(dests, data, opts, true); return req != null? req.getNow(null) : null; } /** * Sends a message to all members and expects responses from members in dests (if non-null). * @param dests A list of group members from which to expect responses (if the call is blocking). * @param data The message to be sent * @param opts A set of options that govern the call. See {@link org.jgroups.blocks.RequestOptions} for details * @return CompletableFuture A future from which the results (RspList) can be retrieved, or null if the request * was sent asynchronously * @throws Exception If the request cannot be sent */ public CompletableFuture> castMessageWithFuture(final Collection
dests, Buffer data, RequestOptions opts) throws Exception { return cast(dests,data,opts,false); } protected GroupRequest cast(final Collection
dests, byte[] data, int offset, int length, RequestOptions options, boolean block_for_results) throws Exception { return cast(dests, new Buffer(data, offset, length), options, block_for_results); } protected GroupRequest cast(final Collection
dests, Buffer data, RequestOptions options, boolean block_for_results) throws Exception { if(options == null) { log.warn("request options were null, using default of sync"); options=RequestOptions.SYNC(); } List
real_dests; // we need to clone because we don't want to modify the original if(dests != null) real_dests=dests.stream().filter(dest -> dest instanceof SiteAddress || this.members.contains(dest)) .collect(ArrayList::new, (list,dest) -> {if(!list.contains(dest)) list.add(dest);}, (l,r) -> {}); else real_dests=new ArrayList<>(members); // Remove the local member from the target destination set if we should not deliver our own message JChannel tmp=channel; if((tmp != null && tmp.getDiscardOwnMessages()) || options.transientFlagSet(Message.TransientFlag.DONT_LOOPBACK)) { if(local_addr == null) local_addr=tmp != null? tmp.getAddress() : null; real_dests.remove(local_addr); } if(options.hasExclusionList()) Stream.of(options.exclusionList()).forEach(real_dests::remove); if(real_dests.isEmpty()) { log.trace("destination list is empty, won't send message"); return empty_group_request; } boolean sync=options.mode() != ResponseMode.GET_NONE; boolean non_blocking=!sync || !block_for_results, anycast=options.anycasting(); if(non_blocking) updateStats(real_dests, anycast, sync, 0); if(!sync) { corr.sendRequest(real_dests, data, null, options); return null; } GroupRequest req=new GroupRequest<>(corr, real_dests, options); long start=non_blocking || !rpc_stats.extendedStats()? 0 : System.nanoTime(); req.execute(data, block_for_results); long time=non_blocking || !rpc_stats.extendedStats()? 0 : System.nanoTime() - start; if(!non_blocking) updateStats(real_dests, anycast, true, time); return req; } public void done(long req_id) { corr.done(req_id); } /** * Sends a unicast message and - depending on the options - returns a result * @param dest the target to which to send the unicast message. Must not be null. * @param data the payload to send * @param offset the offset at which the data starts * @param length the number of bytes to send * @param opts the options to be used * @return T the result. Null if the call is asynchronous (non-blocking) or if the response is null * @throws Exception If there was problem sending the request, processing it at the receiver, or processing * it at the sender. * @throws TimeoutException If the call didn't succeed within the timeout defined in options (if set) */ public T sendMessage(Address dest, byte[] data, int offset, int length, RequestOptions opts) throws Exception { return sendMessage(dest, new Buffer(data, offset, length), opts); } /** * Sends a unicast message and - depending on the options - returns a result * @param dest the target to which to send the unicast message. Must not be null. * @param data the payload to send * @param opts the options to be used * @return T the result. Null if the call is asynchronous (non-blocking) or if the response is null * @throws Exception If there was problem sending the request, processing it at the receiver, or processing * it at the sender. * @throws TimeoutException If the call didn't succeed within the timeout defined in options (if set) */ public T sendMessage(Address dest, Buffer data, RequestOptions opts) throws Exception { if(dest == null) throw new IllegalArgumentException("message destination is null, cannot send message"); if(opts == null) { log.warn("request options were null, using default of sync"); opts=RequestOptions.SYNC(); } // invoke an async RPC directly and return null, without creating a UnicastRequest instance if(opts.mode() == ResponseMode.GET_NONE) { rpc_stats.add(RpcStats.Type.UNICAST, dest, false, 0); corr.sendUnicastRequest(dest, data, null, opts); return null; } // now it must be a sync RPC UnicastRequest req=new UnicastRequest<>(corr, dest, opts); long start=!rpc_stats.extendedStats()? 0 : System.nanoTime(); try { return req.execute(data, true); } finally { long time=!rpc_stats.extendedStats()? 0 : System.nanoTime() - start; rpc_stats.add(RpcStats.Type.UNICAST, dest, true, time); } } /** * Sends a unicast message to the target defined by msg.getDest() and returns a future * @param dest the target to which to send the unicast message. Must not be null. * @param data the payload to send * @param offset the offset at which the data starts * @param length the number of bytes to send * @param opts the options * @return CompletableFuture A future from which the result can be fetched, or null if the call was asynchronous * @throws Exception If there was problem sending the request, processing it at the receiver, or processing * it at the sender. {@link java.util.concurrent.Future#get()} will throw this exception */ public CompletableFuture sendMessageWithFuture(Address dest, byte[] data, int offset, int length, RequestOptions opts) throws Exception { return sendMessageWithFuture(dest, new Buffer(data, offset, length), opts); } /** * Sends a unicast message to the target defined by msg.getDest() and returns a future * @param dest the target to which to send the unicast message. Must not be null. * @param data the payload to send * @param opts the options * @return CompletableFuture A future from which the result can be fetched, or null if the call was asynchronous * @throws Exception If there was problem sending the request, processing it at the receiver, or processing * it at the sender. {@link java.util.concurrent.Future#get()} will throw this exception */ public CompletableFuture sendMessageWithFuture(Address dest, Buffer data, RequestOptions opts) throws Exception { if(dest == null) throw new IllegalArgumentException("message destination is null, cannot send message"); if(opts == null) { log.warn("request options were null, using default of sync"); opts=RequestOptions.SYNC(); } rpc_stats.add(RpcStats.Type.UNICAST, dest, opts.mode() != ResponseMode.GET_NONE, 0); if(opts.mode() == ResponseMode.GET_NONE) { corr.sendUnicastRequest(dest, data, null, opts); return null; } // if we get here, the RPC is synchronous UnicastRequest req=new UnicastRequest<>(corr, dest, opts); req.execute(data, false); return req; } /* ------------------------ RequestHandler Interface ---------------------- */ @Override public Object handle(Message msg) throws Exception { if(req_handler != null) return req_handler.handle(msg); return null; } @Override public void handle(Message request, Response response) throws Exception { if(req_handler != null) { if(async_dispatching) req_handler.handle(request, response); else { Object retval=req_handler.handle(request); if(response != null) response.send(retval, false); } return; } Object retval=handle(request); if(response != null) response.send(retval, false); } /* ------------------ End of RequestHandler Interface----------------- */ protected void updateStats(Collection
dests, boolean anycast, boolean sync, long time) { if(anycast) rpc_stats.addAnycast(sync, time, dests); else rpc_stats.add(RpcStats.Type.MULTICAST, null, sync, time); } protected Object handleUpEvent(Event evt) throws Exception { switch(evt.getType()) { case Event.GET_APPLSTATE: // reply with GET_APPLSTATE_OK byte[] tmp_state=null; if(state_listener != null) { ByteArrayOutputStream output=new ByteArrayOutputStream(1024); state_listener.getState(output); tmp_state=output.toByteArray(); } return new StateTransferInfo(null, 0L, tmp_state); case Event.GET_STATE_OK: if(state_listener != null) { StateTransferResult result=evt.getArg(); if(result.hasBuffer()) { ByteArrayInputStream input=new ByteArrayInputStream(result.getBuffer()); state_listener.setState(input); } } break; case Event.STATE_TRANSFER_OUTPUTSTREAM: OutputStream os=evt.getArg(); if(state_listener != null && os != null) state_listener.getState(os); break; case Event.STATE_TRANSFER_INPUTSTREAM: InputStream is=evt.getArg(); if(state_listener != null && is!=null) state_listener.setState(is); break; case Event.VIEW_CHANGE: View v=evt.getArg(); List
new_mbrs=v.getMembers(); setMembers(new_mbrs); if(membership_listener != null) membership_listener.viewAccepted(v); break; case Event.SET_LOCAL_ADDRESS: log.trace("setting local_addr (%s) to %s", local_addr, evt.getArg()); local_addr=evt.getArg(); break; case Event.SUSPECT: if(membership_listener != null) { // todo: remove in 4.1 once we've completely switched to collections Collection
c=evt.arg() instanceof Address? Collections.singletonList(evt.arg()): evt.arg(); c.forEach(membership_listener::suspect); } break; case Event.BLOCK: if(membership_listener != null) membership_listener.block(); break; case Event.UNBLOCK: if(membership_listener != null) membership_listener.unblock(); break; } return null; } public void channelConnected(JChannel channel) { ; } public void channelDisconnected(JChannel channel) { stop(); } public void channelClosed(JChannel channel) { stop(); } class ProtocolAdapter extends Protocol implements UpHandler { /* ------------------------- Protocol Interface --------------------------- */ @Override public String getName() { return "MessageDispatcher"; } /** * Called by channel (we registered before) when event is received. This is the UpHandler interface. */ @Override public Object up(Event evt) { if(corr != null && !corr.receive(evt)) { try { return handleUpEvent(evt); } catch(Throwable t) { throw new RuntimeException(t); } } return null; } public Object up(Message msg) { if(corr != null) corr.receiveMessage(msg); return null; } public void up(MessageBatch batch) { if(corr == null) return; corr.receiveMessageBatch(batch); } @Override public Object down(Event evt) { return channel != null? channel.down(evt) : null; } public Object down(Message msg) { if(channel != null) { if(!(channel.isConnected() || channel.isConnecting())) { // return null; throw new IllegalStateException("channel is not connected"); } return channel.down(msg); } return null; } /* ----------------------- End of Protocol Interface ------------------------ */ } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy