2018-02-13 12:19:37 +08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
|
|
* accompanied this code).
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License version
|
|
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
|
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
|
|
* questions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import javax.naming.Context;
|
|
|
|
import javax.naming.NamingException;
|
|
|
|
import javax.naming.directory.Attributes;
|
|
|
|
import java.io.PrintStream;
|
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.Path;
|
|
|
|
import java.nio.file.Paths;
|
|
|
|
import java.util.Hashtable;
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* This is utility class for DNS test, it contains many helper methods to
|
|
|
|
* support test construction.
|
|
|
|
*
|
|
|
|
* For basic test sequence see TestBase and DNSTestBase.
|
|
|
|
*/
|
2018-02-13 12:19:37 +08:00
|
|
|
public class DNSTestUtils {
|
|
|
|
public static final String TEST_DNS_SERVER_THREAD = "test.dns.server.thread";
|
|
|
|
public static final String TEST_DNS_ROOT_URL = "test.dns.root.url";
|
|
|
|
public static final int HOSTS_LOOKUP_MAX_DEPTH = 3;
|
|
|
|
|
|
|
|
protected static boolean debug = true;
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Check that attributes contains the mandatory attributes and the right
|
|
|
|
* objectclass attribute.
|
|
|
|
*
|
|
|
|
* @param attrs given attributes to verify
|
|
|
|
* @param mandatory given mandatory for verification
|
|
|
|
* @param optional given optional for verification
|
|
|
|
* @return <tt>true</tt> if check ok
|
2018-02-13 12:19:37 +08:00
|
|
|
*/
|
|
|
|
public static boolean checkSchema(Attributes attrs, String[] mandatory,
|
|
|
|
String[] optional) {
|
|
|
|
// Check mandatory attributes
|
|
|
|
for (String mandatoryAttr : mandatory) {
|
|
|
|
if (attrs.get(mandatoryAttr) == null) {
|
|
|
|
debug("missing mandatory attribute: " + mandatoryAttr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check optional attributes
|
|
|
|
int optMissing = 0;
|
|
|
|
for (String optionalAttr : optional) {
|
|
|
|
if (attrs.get(optionalAttr) == null) {
|
|
|
|
debug("warning: missing optional attribute: " + optionalAttr);
|
|
|
|
++optMissing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attrs.size() > (mandatory.length + (optional.length
|
|
|
|
- optMissing))) {
|
|
|
|
debug("too many attributes: " + attrs);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Process command line arguments and init env.
|
|
|
|
* This method will prepare default environments which to be used to initial
|
|
|
|
* DirContext.
|
|
|
|
*
|
|
|
|
* @param localServer <tt>true</tt> if this test will run against with local
|
|
|
|
* server against dump message playback
|
|
|
|
* @param testname test case name to identify playback file
|
|
|
|
* @param args additional arguments for env preparation
|
|
|
|
* @return prepared env which will be used later to initial DirContext
|
2018-02-13 12:19:37 +08:00
|
|
|
*/
|
2018-07-25 11:03:07 +08:00
|
|
|
public static Hashtable<Object, Object> initEnv(boolean localServer,
|
2018-02-13 12:19:37 +08:00
|
|
|
String testname, String[] args) {
|
|
|
|
|
|
|
|
Hashtable<Object, Object> env = new Hashtable<>();
|
|
|
|
|
|
|
|
// set some default parameters if no additional specified
|
|
|
|
env.put("DNS_DOMAIN", "domain1.com.");
|
|
|
|
env.put("FOREIGN_DOMAIN", "Central.Sun.COM.");
|
|
|
|
env.put("FOREIGN_LEAF", "sunweb");
|
|
|
|
|
|
|
|
// set defaults for some JNDI properties
|
|
|
|
env.put(Context.INITIAL_CONTEXT_FACTORY,
|
|
|
|
"com.sun.jndi.dns.DnsContextFactory");
|
|
|
|
|
|
|
|
boolean traceEnable = false;
|
|
|
|
boolean loopPlayback = false;
|
2018-07-25 11:03:07 +08:00
|
|
|
for (String arg : args) {
|
|
|
|
if (arg.startsWith("-D")) {
|
|
|
|
extractProperty(arg.substring(2), env);
|
|
|
|
} else if (arg.equalsIgnoreCase("-trace")) {
|
2018-02-13 12:19:37 +08:00
|
|
|
traceEnable = true;
|
2018-07-25 11:03:07 +08:00
|
|
|
} else if (arg.equalsIgnoreCase("-loop")) {
|
2018-02-13 12:19:37 +08:00
|
|
|
loopPlayback = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
debug = Boolean.valueOf(System.getProperty("debug", "true"));
|
|
|
|
|
2018-10-22 13:53:39 +08:00
|
|
|
// override testname here if it's been specified
|
|
|
|
String newTestName = (String) env.get("testname");
|
|
|
|
if (newTestName != null && !newTestName.isEmpty()) {
|
|
|
|
testname = newTestName;
|
|
|
|
}
|
|
|
|
|
2018-02-13 12:19:37 +08:00
|
|
|
if (env.get("DNS_SERVER") != null) {
|
|
|
|
String port = (String) env.get("DNS_PORT");
|
|
|
|
String portSuffix = (port == null) ? "" : ":" + port;
|
|
|
|
String url = "dns://" + env.get("DNS_SERVER") + portSuffix;
|
|
|
|
env.put(Context.PROVIDER_URL, url);
|
|
|
|
env.put(Context.PROVIDER_URL, url + "/" + env.get("DNS_DOMAIN"));
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
Thread inst = null;
|
2018-02-13 12:19:37 +08:00
|
|
|
if (traceEnable) {
|
2018-07-25 11:03:07 +08:00
|
|
|
// if trace is enabled, create DNSTracer to dump those message
|
|
|
|
inst = createDNSTracer(testname, env);
|
2018-02-13 12:19:37 +08:00
|
|
|
} else {
|
2018-07-25 11:03:07 +08:00
|
|
|
if (localServer) {
|
|
|
|
// if use local server, create local DNSServer for playback
|
|
|
|
inst = createDNSServer(testname, loopPlayback);
|
2018-02-13 12:19:37 +08:00
|
|
|
} else {
|
|
|
|
// for tests which run against remote server
|
|
|
|
// or no server required
|
2018-07-25 11:03:07 +08:00
|
|
|
debug("Skip local DNS Server creation ");
|
2018-02-13 12:19:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (inst != null) {
|
2018-07-25 11:03:07 +08:00
|
|
|
inst.start();
|
|
|
|
env.put(TEST_DNS_SERVER_THREAD, inst);
|
|
|
|
String url = "dns://localhost:" + ((Server) inst).getPort();
|
2018-02-13 12:19:37 +08:00
|
|
|
|
|
|
|
env.put(TEST_DNS_ROOT_URL, url);
|
|
|
|
env.put(Context.PROVIDER_URL, url + "/" + env.get("DNS_DOMAIN"));
|
|
|
|
}
|
|
|
|
|
|
|
|
return env;
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Clean-up the given directory context.
|
|
|
|
*
|
|
|
|
* @param ctx given context object
|
2018-02-13 12:19:37 +08:00
|
|
|
*/
|
|
|
|
public static void cleanup(Context ctx) {
|
|
|
|
if (ctx != null) {
|
|
|
|
try {
|
|
|
|
ctx.close();
|
|
|
|
} catch (NamingException e) {
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void extractProperty(String propString,
|
|
|
|
Hashtable<Object, Object> env) {
|
|
|
|
int index;
|
|
|
|
|
|
|
|
if ((index = propString.indexOf('=')) > 0) {
|
|
|
|
env.put(propString.substring(0, index),
|
|
|
|
propString.substring(index + 1));
|
|
|
|
} else {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Failed to extract test args property from " + propString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Return new created DNS tracer.
|
|
|
|
*
|
|
|
|
* @param testname test case name to identify playback file
|
|
|
|
* @param env given env for initialization
|
|
|
|
* @return created DNS tracer
|
|
|
|
* @see DNSTracer
|
|
|
|
*/
|
|
|
|
public static DNSTracer createDNSTracer(String testname,
|
|
|
|
Hashtable<Object, Object> env) {
|
2018-02-13 12:19:37 +08:00
|
|
|
try {
|
|
|
|
PrintStream outStream = new PrintStream(getCaptureFile(testname));
|
2018-07-25 11:03:07 +08:00
|
|
|
return new DNSTracer(outStream, (String) env.get("DNS_SERVER"),
|
2018-02-13 12:19:37 +08:00
|
|
|
Integer.parseInt((String) env.get("DNS_PORT")));
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Error: failed to create DNSTracer : " + e.getMessage(), e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Return new created local DNS Server.
|
|
|
|
*
|
|
|
|
* @param testname test case name to identify playback file
|
|
|
|
* @param loop <tt>true</tt> if DNS server required playback message in loop
|
|
|
|
* @return created local DNS Server
|
|
|
|
* @see DNSServer
|
|
|
|
*/
|
|
|
|
public static DNSServer createDNSServer(String testname, boolean loop) {
|
2018-02-13 12:19:37 +08:00
|
|
|
String path = getCaptureFile(testname);
|
|
|
|
if (Files.exists(Paths.get(path))) {
|
2018-07-25 11:03:07 +08:00
|
|
|
try {
|
|
|
|
return new DNSServer(path, loop);
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Error: failed to create DNSServer : " + e.getMessage(),
|
|
|
|
e);
|
|
|
|
}
|
2018-02-13 12:19:37 +08:00
|
|
|
} else {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Error: failed to create DNSServer, not found dns "
|
|
|
|
+ "cache file " + path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Return dns message capture file path.
|
|
|
|
*
|
|
|
|
* @param testname test case name to identify playback file
|
|
|
|
* @return capture file path
|
|
|
|
*/
|
2018-02-13 12:19:37 +08:00
|
|
|
public static String getCaptureFile(String testname) {
|
|
|
|
return Paths.get(System.getProperty("test.src"))
|
|
|
|
.resolve(testname + ".dns").toString();
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Enable hosts file.
|
|
|
|
*
|
|
|
|
* @param hostsFile given hosts file
|
|
|
|
*/
|
2018-02-13 12:19:37 +08:00
|
|
|
public static void enableHostsFile(String hostsFile) {
|
|
|
|
System.out.println("Enable jdk.net.hosts.file = " + hostsFile);
|
|
|
|
System.setProperty("jdk.net.hosts.file", hostsFile);
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Enable hosts file by given searching depth.
|
|
|
|
*
|
|
|
|
* @param depth given depth for searching hosts file
|
|
|
|
*/
|
2018-02-13 12:19:37 +08:00
|
|
|
public static void enableHostsFile(int depth) {
|
|
|
|
Path path = Paths.get(System.getProperty("test.src", "."))
|
|
|
|
.toAbsolutePath();
|
|
|
|
for (int i = depth; i >= 0; i--) {
|
|
|
|
Path filePath = path.resolve("hosts");
|
|
|
|
if (Files.exists(filePath) && !Files.isDirectory(filePath)) {
|
|
|
|
enableHostsFile(filePath.toString());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
path = path.getParent();
|
|
|
|
if (path == null) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Print given object if debug enabled.
|
|
|
|
*
|
|
|
|
* @param object given object to print
|
|
|
|
*/
|
2018-02-13 12:19:37 +08:00
|
|
|
public static void debug(Object object) {
|
|
|
|
if (debug) {
|
|
|
|
System.out.println(object);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 11:03:07 +08:00
|
|
|
/**
|
|
|
|
* Verify attributes contains the mandatory attributes and the right
|
|
|
|
* objectclass attribute, will throw RuntimeException if verify failed.
|
|
|
|
*
|
|
|
|
* @param attrs given attributes to verify
|
|
|
|
* @param mandatory given mandatory for verification
|
|
|
|
* @param optional given optional for verification
|
|
|
|
*/
|
2018-02-13 12:19:37 +08:00
|
|
|
public static void verifySchema(Attributes attrs, String[] mandatory,
|
|
|
|
String[] optional) {
|
|
|
|
debug(attrs);
|
|
|
|
if (!checkSchema(attrs, mandatory, optional)) {
|
|
|
|
throw new RuntimeException("Check schema failed.");
|
|
|
|
}
|
|
|
|
}
|
2018-07-25 11:03:07 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return dns root url.
|
|
|
|
*
|
|
|
|
* @param env given env
|
|
|
|
* @return dns root url
|
|
|
|
*/
|
|
|
|
public static String getRootUrl(Hashtable<Object, Object> env) {
|
|
|
|
return (String) env.get(TEST_DNS_ROOT_URL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assemble a fully-qualified domain name from the base component and the
|
|
|
|
* domain name.
|
|
|
|
*
|
|
|
|
* @param base given base component
|
|
|
|
* @param env given env
|
|
|
|
* @param primary <tt>true</tt> if primary domain
|
|
|
|
* @return assembled fully-qualified domain name
|
|
|
|
*/
|
|
|
|
public static String buildFqdn(String base, Hashtable<Object, Object> env,
|
|
|
|
boolean primary) {
|
|
|
|
String domain = (String) (primary ?
|
|
|
|
env.get("DNS_DOMAIN") :
|
|
|
|
env.get("FOREIGN_DOMAIN"));
|
|
|
|
|
|
|
|
return base + "." + domain;
|
|
|
|
}
|
2018-02-13 12:19:37 +08:00
|
|
|
}
|