org.jgroups.protocols.BPING Maven / Gradle / Ivy
package org.jgroups.protocols;
import org.jgroups.BytesMessage;
import org.jgroups.Message;
import org.jgroups.annotations.Property;
import org.jgroups.util.ByteArrayDataInputStream;
import org.jgroups.util.ByteArrayDataOutputStream;
import org.jgroups.util.Util;
import java.io.DataInput;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
/**
* Broadcast PING. Uses UDP broadcasts to discover initial membership. This protocol is useless in IPv6 environments, as
* IPv6 has no notion of broadcast addresses. Use IP multicasts instead (e.g. PING or MPING) when running in IPv6.
* @author Bela Ban
* @since 2.12
*/
public class BPING extends PING implements Runnable {
/* ----------------------------------------- Properties -------------------------------------------------- */
@Property(description="Target address for broadcasts. This should be restricted to the local subnet, e.g. 192.168.1.255")
protected String dest="255.255.255.255";
@Property(description="Port for discovery packets")
protected int bind_port=8555;
@Property(description="Sends discovery packets to ports 8555 to (8555+port_range)")
protected int port_range=5;
/* --------------------------------------------- Fields ------------------------------------------------------ */
protected DatagramSocket sock;
protected volatile Thread receiver;
protected InetAddress dest_addr;
public BPING() {
}
public int getBindPort() {
return bind_port;
}
public BPING setBindPort(int bind_port) {
this.bind_port=bind_port; return this;
}
public void init() throws Exception {
super.init();
dest_addr=InetAddress.getByName(dest);
if(log.isDebugEnabled())
log.debug("listening on " + bind_port);
}
public void start() throws Exception {
for(int i=bind_port; i <= bind_port+port_range; i++) {
try {
sock=getSocketFactory().createDatagramSocket("jgroups.bping.sock", i);
break;
}
catch(Throwable t) {
if(i > bind_port+port_range)
throw new RuntimeException("failed to open a port in range [" + bind_port + " - " + (bind_port+port_range) + "]", t);
}
}
if (null == sock)
throw new RuntimeException("failed to open a port in range [" + bind_port + " - " + (bind_port+port_range) + "]");
sock.setBroadcast(true);
startReceiver();
super.start();
}
private void startReceiver() {
if(receiver == null || !receiver.isAlive()) {
receiver=new Thread(this, "ReceiverThread");
receiver.setDaemon(true);
receiver.start();
log.trace("receiver thread started");
}
}
public void stop() {
Util.close(sock);
sock=null;
receiver=null;
super.stop();
}
@Override
protected void sendMcastDiscoveryRequest(Message msg) {
try {
if(msg.getSrc() == null)
msg.setSrc(local_addr);
ByteArrayDataOutputStream out=new ByteArrayDataOutputStream(msg.size());
msg.writeTo(out);
for(int i=bind_port; i <= bind_port+port_range; i++) {
DatagramPacket packet=new DatagramPacket(out.buffer(), 0, out.position(), dest_addr, i);
sock.send(packet);
}
}
catch(Exception ex) {
log.error(String.format("%s: failed sending discovery request", local_addr), ex);
}
}
public void run() {
final byte[] receive_buf=new byte[65535];
DatagramPacket packet=new DatagramPacket(receive_buf, receive_buf.length);
DataInput inp;
while(sock != null && Thread.currentThread().equals(receiver)) {
packet.setData(receive_buf, 0, receive_buf.length);
try {
sock.receive(packet);
inp=new ByteArrayDataInputStream(packet.getData(), packet.getOffset(), packet.getLength());
Message msg=new BytesMessage();
msg.readFrom(inp);
up(msg);
}
catch(SocketException socketEx) {
break;
}
catch(Throwable ex) {
log.error(Util.getMessage("FailedReceivingPacketFrom"), packet.getSocketAddress(), ex);
}
}
if(log.isTraceEnabled())
log.trace("receiver thread terminated");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy