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

org.jgroups.protocols.dns.DNS_PING 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.protocols.dns;

import org.jgroups.Address;
import org.jgroups.Event;
import org.jgroups.Message;
import org.jgroups.PhysicalAddress;
import org.jgroups.annotations.ManagedOperation;
import org.jgroups.annotations.Property;
import org.jgroups.protocols.Discovery;
import org.jgroups.protocols.PingData;
import org.jgroups.protocols.PingHeader;
import org.jgroups.stack.IpAddress;
import org.jgroups.util.Buffer;
import org.jgroups.util.NameCache;
import org.jgroups.util.Responses;

import java.util.ArrayList;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;

import static org.jgroups.Message.Flag.*;
import static org.jgroups.Message.TransientFlag.DONT_LOOPBACK;

public class DNS_PING extends Discovery {

    private static final String DEFAULT_DNS_FACTORY = "com.sun.jndi.dns.DnsContextFactory";
    private static final String DEFAULT_DNS_RECORD_TYPE = "A";

    @Property(description = "DNS Context Factory.  Used when DNS_PING is configured to use SRV record types and when using A types with a specific dns_address.")
    protected String  dns_context_factory = DEFAULT_DNS_FACTORY;

    @Property(description = "DNS Address. This property will be assembled with the 'dns://' prefix.  If this is specified, A records will be resolved through DnsContext.")
    protected String  dns_address = "";

    @Property(description = "DNS Record type")
    protected String  dns_record_type = DEFAULT_DNS_RECORD_TYPE;

    @Property(description = "A comma-separated list of DNS queries for fetching members")
    protected String  dns_query;

    @Property(description="For SRV records returned by the DNS query, the non-0 ports returned by DNS are" +
      "used. If this attribute is true, then the transport ports will also be used. Ignored for A records.")
    protected boolean probe_transport_ports;



    protected volatile DNSResolver dns_resolver;

    private int                    transportPort, portRange;

    @Override
    public void init() throws Exception {
        super.init();
        validateProperties();
        transportPort = getTransport().getBindPort();
        portRange = getTransport().getPortRange();
        if (dns_resolver == null) {
            if (dns_address == null || dns_address.isEmpty()) {
                dns_resolver = new DefaultDNSResolver(dns_context_factory, dns_address);
            } else {
                dns_resolver = new AddressedDNSResolver(dns_context_factory, dns_address);
            }
        }
    }

    protected void validateProperties() {
        if (dns_query == null || dns_query.trim().isEmpty()) {
            throw new IllegalArgumentException("dns_query can not be null or empty");
        }
    }

    @Override
    public void destroy() {
        if (dns_resolver != null) {
            dns_resolver.close();
        }
    }

    @Override
    public boolean isDynamic() {
        return true;
    }

    /**
     * Gets a list of IP addresses for the provided DNS query list and record type.
     *
     * @param dns_query A comma-separated list of DNS queries. Must not be {@code null}.
     * @param dns_record_type The DNS record type.
     *
     * @return A list of IP addresses corresponding to the provided DNS query list and record type. An empty list is
     * returned if the provided DNS query does not resolve to any IP addresses.
     *
     * @throws NullPointerException if dns_query is {@code null}.
     */
    List
getMembers(final String dns_query, final DNSResolver.DNSRecordType dns_record_type) { final List
dns_discovery_members = new ArrayList<>(); for (final String query : dns_query.split(",")) { final List
addresses = dns_resolver.resolveIps(query.trim(), dns_record_type); if (addresses != null) { dns_discovery_members.addAll(addresses); } } return dns_discovery_members; } @ManagedOperation(description="Executes the DNS query and returns the result in string format") public String fetchFromDns() { long start=System.currentTimeMillis(); List
dns_discovery_members = getMembers(dns_query, DNSResolver.DNSRecordType.valueOf(dns_record_type)); String ret=dns_discovery_members != null && !dns_discovery_members.isEmpty() ? dns_discovery_members.toString() : null; long time=System.currentTimeMillis()-start; return String.format("%s\n(took %d ms)\n", ret, time); } @Override public void findMembers(List
members, boolean initial_discovery, Responses responses) { PingData data = null; PhysicalAddress physical_addr = null; Set cluster_members=new LinkedHashSet<>(); DNSResolver.DNSRecordType record_type=DNSResolver.DNSRecordType.valueOf(dns_record_type); if (!use_ip_addrs || !initial_discovery) { physical_addr = (PhysicalAddress) down(new Event(Event.GET_PHYSICAL_ADDRESS, local_addr)); // https://issues.jboss.org/browse/JGRP-1670 data = new PingData(local_addr, false, NameCache.get(local_addr), physical_addr); if (members != null && members.size() <= max_members_in_discovery_request) data.mbrs(members); } long start=System.currentTimeMillis(); List
dns_discovery_members = getMembers(dns_query, record_type); long time=System.currentTimeMillis()-start; if(log.isDebugEnabled()) { if(dns_discovery_members != null && !dns_discovery_members.isEmpty()) log.debug("%s: entries collected from DNS (in %d ms): %s", local_addr, time, dns_discovery_members); else log.debug("%s: no entries collected from DNS (in %d ms)", local_addr, time); } boolean ports_found=false; if (dns_discovery_members != null) { for (Address address : dns_discovery_members) { if (address.equals(physical_addr)) // no need to send the request to myself continue; if(address instanceof IpAddress) { IpAddress ip = ((IpAddress) address); if(record_type == DNSResolver.DNSRecordType.SRV && ip.getPort() > 0) { ports_found=true; cluster_members.add(ip); if(!probe_transport_ports) continue; } for(int i=0; i <= portRange; i++) cluster_members.add(new IpAddress(ip.getIpAddress(), transportPort + i)); } } } if(dns_discovery_members != null && !dns_discovery_members.isEmpty() && log.isDebugEnabled()) { if(ports_found) log.debug("%s: sending discovery requests to %s", local_addr, cluster_members); else log.debug("%s: sending discovery requests to hosts %s on ports [%d .. %d]", local_addr, dns_discovery_members, transportPort, transportPort + portRange); } Buffer data_buf=data != null? marshal(data) : null; PingHeader hdr = new PingHeader(PingHeader.GET_MBRS_REQ).clusterName(cluster_name).initialDiscovery(initial_discovery); for (Address addr: cluster_members) { // the message needs to be DONT_BUNDLE, see explanation above Message msg = new Message(addr).setFlag(INTERNAL, DONT_BUNDLE, OOB).setTransientFlag(DONT_LOOPBACK) .putHeader(this.id, hdr); if (data_buf != null) msg.setBuffer(data_buf); if (async_discovery_use_separate_thread_per_request) timer.execute(() -> sendDiscoveryRequest(msg), sends_can_block); else sendDiscoveryRequest(msg); } } protected void sendDiscoveryRequest(Message req) { try { log.trace("%s: sending discovery request to %s", local_addr, req.getDest()); down_prot.down(req); } catch (Throwable t) { log.error("sending discovery request to %s failed: %s", req.dest(), t); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy