Added a DNSPacket class and Initial implementation of custom BinaryFields
This commit is contained in:
parent
609716adcb
commit
6071ad7c70
7 changed files with 171 additions and 27 deletions
52
src/zutil/net/dns/DNSPacket.java
Executable file
52
src/zutil/net/dns/DNSPacket.java
Executable file
|
|
@ -0,0 +1,52 @@
|
||||||
|
package zutil.net.dns;
|
||||||
|
|
||||||
|
import zutil.parser.binary.BinaryStructOutputStream;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is a general wrapper for a whole DNS packet.
|
||||||
|
*
|
||||||
|
* Created by ezivkoc on 2016-04-11.
|
||||||
|
*/
|
||||||
|
public class DNSPacket {
|
||||||
|
private DNSPacketHeader header;
|
||||||
|
private ArrayList<DNSPacketQuestion> questions;
|
||||||
|
private ArrayList<DNSPacketResource> answerRecords;
|
||||||
|
private ArrayList<DNSPacketResource> nameServers;
|
||||||
|
private ArrayList<DNSPacketResource> additionalRecords;
|
||||||
|
|
||||||
|
|
||||||
|
public DNSPacket(){
|
||||||
|
header = new DNSPacketHeader();
|
||||||
|
questions = new ArrayList<>();
|
||||||
|
answerRecords = new ArrayList<>();
|
||||||
|
nameServers = new ArrayList<>();
|
||||||
|
additionalRecords = new ArrayList<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static DNSPacket read(InputStream in){
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void write(OutputStream out) throws IOException {
|
||||||
|
BinaryStructOutputStream structOut = new BinaryStructOutputStream(out);
|
||||||
|
structOut.write(header);
|
||||||
|
out.flush();
|
||||||
|
|
||||||
|
/*for (DNSPacketQuestion question : questions)
|
||||||
|
question.write(out);
|
||||||
|
for (DNSPacketResource answerRecord : answerRecords)
|
||||||
|
answerRecord.write(out);
|
||||||
|
for (DNSPacketResource nameServer : nameServers)
|
||||||
|
nameServer.write(out);
|
||||||
|
for (DNSPacketResource additionalRecord : additionalRecords)
|
||||||
|
additionalRecord.write(out);*/
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -24,8 +24,14 @@
|
||||||
|
|
||||||
package zutil.net.dns;
|
package zutil.net.dns;
|
||||||
|
|
||||||
|
import zutil.parser.binary.BinaryFieldData;
|
||||||
|
import zutil.parser.binary.BinaryFieldSerializer;
|
||||||
import zutil.parser.binary.BinaryStruct;
|
import zutil.parser.binary.BinaryStruct;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Created by Ziver on 2016-02-09.
|
* Created by Ziver on 2016-02-09.
|
||||||
* Reference: http://tools.ietf.org/html/rfc1035
|
* Reference: http://tools.ietf.org/html/rfc1035
|
||||||
|
|
@ -64,6 +70,40 @@ public class DNSPacketQuestion implements BinaryStruct {
|
||||||
For example, the QCLASS field is IN for the Internet.
|
For example, the QCLASS field is IN for the Internet.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@CustomBinaryField(index=10, serializer=QNameSerializer.class)
|
||||||
|
private String qName;
|
||||||
|
@BinaryField(index=10, length=1)
|
||||||
|
private int qType;
|
||||||
|
@BinaryField(index=20, length=1)
|
||||||
|
private int qClass;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private static class QNameSerializer implements BinaryFieldSerializer<String> {
|
||||||
|
|
||||||
|
public String read(InputStream in, BinaryFieldData field) throws IOException {
|
||||||
|
StringBuilder str = new StringBuilder();
|
||||||
|
int c = in.read();
|
||||||
|
while (c > 0){
|
||||||
|
for (int i=0; i<c; ++i){
|
||||||
|
str.append((char)in.read());
|
||||||
|
}
|
||||||
|
c = in.read();
|
||||||
|
if (c > 0)
|
||||||
|
str.append('.');
|
||||||
|
}
|
||||||
|
return toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void write(OutputStream out, String domain, BinaryFieldData field) throws IOException {
|
||||||
|
String[] labels = domain.split(".");
|
||||||
|
for (String label : labels){
|
||||||
|
out.write(label.length());
|
||||||
|
out.write(label.getBytes());
|
||||||
|
}
|
||||||
|
out.write(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -26,6 +26,10 @@ package zutil.net.dns;
|
||||||
|
|
||||||
import zutil.parser.binary.BinaryStruct;
|
import zutil.parser.binary.BinaryStruct;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Created by Ziver on 2016-02-09.
|
* Created by Ziver on 2016-02-09.
|
||||||
* Reference: http://tools.ietf.org/html/rfc1035
|
* Reference: http://tools.ietf.org/html/rfc1035
|
||||||
|
|
@ -85,4 +89,13 @@ public class DNSPacketResource implements BinaryStruct {
|
||||||
the RDATA field is a 4 octet ARPA Internet address.
|
the RDATA field is a 4 octet ARPA Internet address.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
public static DNSPacketResource read(InputStream in){
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void write(OutputStream out) throws IOException {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -53,11 +53,10 @@ public class MulticastDNSClient extends ThreadedUDPNetwork implements ThreadedUD
|
||||||
BinaryStructOutputStream out = new BinaryStructOutputStream(buffer);
|
BinaryStructOutputStream out = new BinaryStructOutputStream(buffer);
|
||||||
|
|
||||||
DNSPacketHeader header = new DNSPacketHeader();
|
DNSPacketHeader header = new DNSPacketHeader();
|
||||||
|
header.setDefaultQueryData();
|
||||||
out.write(header);
|
out.write(header);
|
||||||
|
|
||||||
DatagramPacket packet = null;
|
DatagramPacket packet = new DatagramPacket(
|
||||||
|
|
||||||
packet = new DatagramPacket(
|
|
||||||
buffer.toByteArray(), buffer.size(),
|
buffer.toByteArray(), buffer.size(),
|
||||||
InetAddress.getByName( MDNS_MULTICAST_ADDR ),
|
InetAddress.getByName( MDNS_MULTICAST_ADDR ),
|
||||||
MDNS_MULTICAST_PORT );
|
MDNS_MULTICAST_PORT );
|
||||||
|
|
|
||||||
|
|
@ -1,48 +1,62 @@
|
||||||
package zutil.parser.binary;
|
package zutil.parser.binary;
|
||||||
|
|
||||||
|
|
||||||
import java.io.InvalidClassException;
|
|
||||||
import java.io.InvalidObjectException;
|
|
||||||
import java.lang.reflect.Field;
|
import java.lang.reflect.Field;
|
||||||
import java.util.ArrayList;
|
import java.util.*;
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import zutil.ByteUtil;
|
|
||||||
import zutil.converter.Converter;
|
import zutil.converter.Converter;
|
||||||
import zutil.parser.binary.BinaryStruct.BinaryField;
|
import zutil.parser.binary.BinaryStruct.*;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A class representing each field in a BinaryStruct.
|
* A class representing each field in a BinaryStruct.
|
||||||
*/
|
*/
|
||||||
public class BinaryFieldData implements Comparable<BinaryFieldData> {
|
public class BinaryFieldData {
|
||||||
private static final HashMap<Class, List<BinaryFieldData>> cache = new HashMap<>();
|
private static final HashMap<Class, List<BinaryFieldData>> cache = new HashMap<>();
|
||||||
|
|
||||||
private int index;
|
private int index;
|
||||||
private int length;
|
private int length;
|
||||||
|
private BinaryFieldSerializer serializer;
|
||||||
private Field field;
|
private Field field;
|
||||||
|
|
||||||
|
|
||||||
protected static List<BinaryFieldData> getStructFieldList(Class<? extends BinaryStruct> clazz){
|
protected static List<BinaryFieldData> getStructFieldList(Class<? extends BinaryStruct> clazz){
|
||||||
if (!cache.containsKey(clazz)) {
|
if (!cache.containsKey(clazz)) {
|
||||||
ArrayList<BinaryFieldData> list = new ArrayList<>();
|
try {
|
||||||
for (Field field : clazz.getDeclaredFields()) {
|
ArrayList<BinaryFieldData> list = new ArrayList<>();
|
||||||
if (field.isAnnotationPresent(BinaryField.class))
|
for (Field field : clazz.getDeclaredFields()) {
|
||||||
list.add(new BinaryFieldData(field));
|
if (field.isAnnotationPresent(BinaryField.class) ||
|
||||||
|
field.isAnnotationPresent(CustomBinaryField.class))
|
||||||
|
|
||||||
|
list.add(new BinaryFieldData(field));
|
||||||
|
|
||||||
|
}
|
||||||
|
Collections.sort(list, new Comparator<BinaryFieldData>(){
|
||||||
|
@Override
|
||||||
|
public int compare(BinaryFieldData o1, BinaryFieldData o2) {
|
||||||
|
return o1.index - o2.index;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
cache.put(clazz, list);
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
Collections.sort(list);
|
|
||||||
cache.put(clazz, list);
|
|
||||||
}
|
}
|
||||||
return cache.get(clazz);
|
return cache.get(clazz);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private BinaryFieldData(Field f){
|
private BinaryFieldData(Field f) throws IllegalAccessException, InstantiationException {
|
||||||
field = f;
|
field = f;
|
||||||
BinaryField fieldData = field.getAnnotation(BinaryField.class);
|
if (field.isAnnotationPresent(CustomBinaryField.class)){
|
||||||
index = fieldData.index();
|
CustomBinaryField fieldData = field.getAnnotation(CustomBinaryField.class);
|
||||||
length = fieldData.length();
|
index = fieldData.index();
|
||||||
|
serializer = (BinaryFieldSerializer) fieldData.serializer().newInstance();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
BinaryField fieldData = field.getAnnotation(BinaryField.class);
|
||||||
|
index = fieldData.index();
|
||||||
|
length = fieldData.length();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void setValue(Object obj, byte[] data){
|
protected void setValue(Object obj, byte[] data){
|
||||||
|
|
@ -54,6 +68,8 @@ public class BinaryFieldData implements Comparable<BinaryFieldData> {
|
||||||
field.set(obj, Converter.toInt(data));
|
field.set(obj, Converter.toInt(data));
|
||||||
else if (field.getType() == String.class)
|
else if (field.getType() == String.class)
|
||||||
field.set(obj, new String(data));
|
field.set(obj, new String(data));
|
||||||
|
else
|
||||||
|
throw new UnsupportedOperationException("Unsupported BinaryStruct field class: "+ field.getClass());
|
||||||
} catch (IllegalAccessException e){
|
} catch (IllegalAccessException e){
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
|
|
@ -81,8 +97,4 @@ public class BinaryFieldData implements Comparable<BinaryFieldData> {
|
||||||
return length;
|
return length;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public int compareTo(BinaryFieldData o) {
|
|
||||||
return this.index - o.index;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
21
src/zutil/parser/binary/BinaryFieldSerializer.java
Executable file
21
src/zutil/parser/binary/BinaryFieldSerializer.java
Executable file
|
|
@ -0,0 +1,21 @@
|
||||||
|
package zutil.parser.binary;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An Interface where custom field parser and writer can be implemented.
|
||||||
|
* NOTE: Partial octet serializing not supported.
|
||||||
|
*
|
||||||
|
* Created by Ziver on 2016-04-11.
|
||||||
|
*/
|
||||||
|
public interface BinaryFieldSerializer<T> {
|
||||||
|
|
||||||
|
T read(InputStream in,
|
||||||
|
BinaryFieldData field) throws IOException;
|
||||||
|
|
||||||
|
void write(OutputStream out,
|
||||||
|
T obj,
|
||||||
|
BinaryFieldData field) throws IOException;
|
||||||
|
}
|
||||||
|
|
@ -30,7 +30,7 @@ import java.lang.annotation.RetentionPolicy;
|
||||||
import java.lang.annotation.Target;
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Created by ezivkoc on 2016-01-28.
|
* Created by Ziver on 2016-01-28.
|
||||||
*/
|
*/
|
||||||
public interface BinaryStruct {
|
public interface BinaryStruct {
|
||||||
|
|
||||||
|
|
@ -40,4 +40,11 @@ public interface BinaryStruct {
|
||||||
int index();
|
int index();
|
||||||
int length();
|
int length();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target(ElementType.FIELD)
|
||||||
|
@interface CustomBinaryField{
|
||||||
|
int index();
|
||||||
|
Class serializer();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue