blob: 520ff53f7a1ae0c9c9f776e3fae9e89d773e8107 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.log4j.net;
import org.apache.logging.log4j.status.StatusLogger;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;
public class ZeroConfSupport {
private static Object jmDNS = initializeJMDNS();
private static Class jmDNSClass;
private static Class serviceInfoClass;
private Object serviceInfo;
public ZeroConfSupport(String zone, int port, String name, Map properties) {
//if version 3 is available, use it to construct a serviceInfo instance, otherwise support the version1 API
boolean isVersion3 = false;
try {
//create method is in version 3, not version 1
jmDNSClass.getMethod("create", null);
isVersion3 = true;
} catch (NoSuchMethodException e) {
//no-op
}
if (isVersion3) {
StatusLogger.getLogger().debug("using JmDNS version 3 to construct serviceInfo instance");
serviceInfo = buildServiceInfoVersion3(zone, port, name, properties);
} else {
StatusLogger.getLogger().debug("using JmDNS version 1.0 to construct serviceInfo instance");
serviceInfo = buildServiceInfoVersion1(zone, port, name, properties);
}
}
public ZeroConfSupport(String zone, int port, String name) {
this(zone, port, name, new HashMap());
}
private static Object createJmDNSVersion1() {
try {
return jmDNSClass.newInstance();
} catch (InstantiationException | IllegalAccessException e) {
StatusLogger.getLogger().warn("Unable to instantiate JMDNS", e);
}
return null;
}
private static Object createJmDNSVersion3() {
try {
Method jmDNSCreateMethod = jmDNSClass.getMethod("create", null);
return jmDNSCreateMethod.invoke(null, null);
} catch (IllegalAccessException e) {
StatusLogger.getLogger().warn("Unable to instantiate jmdns class", e);
} catch (NoSuchMethodException e) {
StatusLogger.getLogger().warn("Unable to access constructor", e);
} catch (InvocationTargetException e) {
StatusLogger.getLogger().warn("Unable to call constructor", e);
}
return null;
}
private Object buildServiceInfoVersion1(String zone, int port, String name, Map properties) {
//version 1 uses a hashtable
Hashtable hashtableProperties = new Hashtable(properties);
try {
Class[] args = new Class[6];
args[0] = String.class;
args[1] = String.class;
args[2] = int.class;
args[3] = int.class; //weight (0)
args[4] = int.class; //priority (0)
args[5] = Hashtable.class;
Constructor constructor = serviceInfoClass.getConstructor(args);
Object[] values = new Object[6];
values[0] = zone;
values[1] = name;
values[2] = port;
values[3] = 0;
values[4] = 0;
values[5] = hashtableProperties;
Object result = constructor.newInstance(values);
StatusLogger.getLogger().debug("created serviceinfo: " + result);
return result;
} catch (IllegalAccessException | InstantiationException | InvocationTargetException e) {
StatusLogger.getLogger().warn("Unable to construct ServiceInfo instance", e);
} catch (NoSuchMethodException e) {
StatusLogger.getLogger().warn("Unable to get ServiceInfo constructor", e);
}
return null;
}
private Object buildServiceInfoVersion3(String zone, int port, String name, Map properties) {
try {
Class[] args = new Class[6];
args[0] = String.class; //zone/type
args[1] = String.class; //display name
args[2] = int.class; //port
args[3] = int.class; //weight (0)
args[4] = int.class; //priority (0)
args[5] = Map.class;
Method serviceInfoCreateMethod = serviceInfoClass.getMethod("create", args);
Object[] values = new Object[6];
values[0] = zone;
values[1] = name;
values[2] = port;
values[3] = 0;
values[4] = 0;
values[5] = properties;
Object result = serviceInfoCreateMethod.invoke(null, values);
StatusLogger.getLogger().debug("created serviceinfo: " + result);
return result;
} catch (IllegalAccessException | InvocationTargetException e) {
StatusLogger.getLogger().warn("Unable to invoke create method", e);
} catch (NoSuchMethodException e) {
StatusLogger.getLogger().warn("Unable to find create method", e);
}
return null;
}
public void advertise() {
try {
Method method = jmDNSClass.getMethod("registerService", new Class[] {serviceInfoClass});
method.invoke(jmDNS, serviceInfo);
StatusLogger.getLogger().debug("registered serviceInfo: " + serviceInfo);
} catch (IllegalAccessException | InvocationTargetException e) {
StatusLogger.getLogger().warn("Unable to invoke registerService method", e);
} catch (NoSuchMethodException e) {
StatusLogger.getLogger().warn("No registerService method", e);
}
}
public void unadvertise() {
try {
Method method = jmDNSClass.getMethod("unregisterService", serviceInfoClass);
method.invoke(jmDNS, serviceInfo);
StatusLogger.getLogger().debug("unregistered serviceInfo: " + serviceInfo);
} catch (IllegalAccessException | InvocationTargetException e) {
StatusLogger.getLogger().warn("Unable to invoke unregisterService method", e);
} catch (NoSuchMethodException e) {
StatusLogger.getLogger().warn("No unregisterService method", e);
}
}
private static Object initializeJMDNS() {
try {
jmDNSClass = Class.forName("javax.jmdns.JmDNS");
serviceInfoClass = Class.forName("javax.jmdns.ServiceInfo");
} catch (ClassNotFoundException e) {
StatusLogger.getLogger().warn("JmDNS or serviceInfo class not found", e);
}
//if version 3 is available, use it to construct a serviceInfo instance, otherwise support the version1 API
boolean isVersion3 = false;
try {
//create method is in version 3, not version 1
jmDNSClass.getMethod("create", null);
isVersion3 = true;
} catch (NoSuchMethodException e) {
//no-op
}
if (isVersion3) {
return createJmDNSVersion3();
} else {
return createJmDNSVersion1();
}
}
public static Object getJMDNSInstance() {
return jmDNS;
}
}