/* * Copied from the DnsJava project * * Copyright (c) 1998-2011, Brian Wellington. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ package io.milton.dns.tools; import java.io.*; import java.net.*; import io.milton.dns.*; import io.milton.dns.record.DClass; import io.milton.dns.record.ExtendedFlags; import io.milton.dns.record.Message; import io.milton.dns.record.Rcode; import io.milton.dns.record.Record; import io.milton.dns.record.ReverseMap; import io.milton.dns.record.Section; import io.milton.dns.record.SimpleResolver; import io.milton.dns.record.TSIG; import io.milton.dns.record.Type; /** * @author Brian Wellington <bwelling@xbill.org> */ public class dig { static Name name = null; static int type = Type.A, dclass = DClass.IN; static void usage() { System.out.println("Usage: dig [@server] name [<type>] [<class>] [options]"); System.exit(0); } static void doQuery(Message response, long ms) throws IOException { System.out.println("; java dig 0.0"); System.out.println(response); System.out.println(";; Query time: " + ms + " ms"); } static void doAXFR(Message response) throws IOException { System.out.println("; java dig 0.0 <> " + name + " axfr"); if (response.isSigned()) { System.out.print(";; TSIG "); if (response.isVerified()) { System.out.println("ok"); } else { System.out.println("failed"); } } if (response.getRcode() != Rcode.NOERROR) { System.out.println(response); return; } Record[] records = response.getSectionArray(Section.ANSWER); for (int i = 0; i < records.length; i++) { System.out.println(records[i]); } System.out.print(";; done ("); System.out.print(response.getHeader().getCount(Section.ANSWER)); System.out.print(" records, "); System.out.print(response.getHeader().getCount(Section.ADDITIONAL)); System.out.println(" additional)"); } public static void main(String argv[]) throws IOException { String server = null; int arg; Message query, response; Record rec; SimpleResolver res = null; boolean printQuery = false; long startTime, endTime; if (argv.length < 1) { usage(); } try { arg = 0; if (argv[arg].startsWith("@")) { server = argv[arg++].substring(1); } if (server != null) { res = new SimpleResolver(server); } else { res = new SimpleResolver(); } String nameString = argv[arg++]; if (nameString.equals("-x")) { name = ReverseMap.fromAddress(argv[arg++]); type = Type.PTR; dclass = DClass.IN; } else { name = Name.fromString(nameString, Name.root); type = Type.value(argv[arg]); if (type < 0) { type = Type.A; } else { arg++; } dclass = DClass.value(argv[arg]); if (dclass < 0) { dclass = DClass.IN; } else { arg++; } } while (argv[arg].startsWith("-") && argv[arg].length() > 1) { switch (argv[arg].charAt(1)) { case 'p': String portStr; int port; if (argv[arg].length() > 2) { portStr = argv[arg].substring(2); } else { portStr = argv[++arg]; } port = Integer.parseInt(portStr); if (port < 0 || port > 65536) { System.out.println("Invalid port"); return; } res.setPort(port); break; case 'b': String addrStr; if (argv[arg].length() > 2) { addrStr = argv[arg].substring(2); } else { addrStr = argv[++arg]; } InetAddress addr; try { addr = InetAddress.getByName(addrStr); } catch (Exception e) { System.out.println("Invalid address"); return; } res.setLocalAddress(addr); break; case 'k': String key; if (argv[arg].length() > 2) { key = argv[arg].substring(2); } else { key = argv[++arg]; } res.setTSIGKey(TSIG.fromString(key)); break; case 't': res.setTCP(true); break; case 'i': res.setIgnoreTruncation(true); break; case 'e': String ednsStr; int edns; if (argv[arg].length() > 2) { ednsStr = argv[arg].substring(2); } else { ednsStr = argv[++arg]; } edns = Integer.parseInt(ednsStr); if (edns < 0 || edns > 1) { System.out.println("Unsupported " + "EDNS level: " + edns); return; } res.setEDNS(edns); break; case 'd': res.setEDNS(0, 0, ExtendedFlags.DO, null); break; case 'q': printQuery = true; break; default: System.out.print("Invalid option: "); System.out.println(argv[arg]); } arg++; } } catch (ArrayIndexOutOfBoundsException e) { if (name == null) { usage(); } } if (res == null) { res = new SimpleResolver(); } rec = Record.newRecord(name, type, dclass); query = Message.newQuery(rec); if (printQuery) { System.out.println(query); } startTime = System.currentTimeMillis(); response = res.send(query); endTime = System.currentTimeMillis(); if (type == Type.AXFR) { doAXFR(response); } else { doQuery(response, endTime - startTime); } } }