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;
|
||||
|
||||
import zutil.parser.binary.BinaryFieldData;
|
||||
import zutil.parser.binary.BinaryFieldSerializer;
|
||||
import zutil.parser.binary.BinaryStruct;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.OutputStream;
|
||||
|
||||
/**
|
||||
* Created by Ziver on 2016-02-09.
|
||||
* 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.
|
||||
*/
|
||||
|
||||
@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 java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.OutputStream;
|
||||
|
||||
/**
|
||||
* Created by Ziver on 2016-02-09.
|
||||
* 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.
|
||||
*/
|
||||
|
||||
|
||||
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);
|
||||
|
||||
DNSPacketHeader header = new DNSPacketHeader();
|
||||
header.setDefaultQueryData();
|
||||
out.write(header);
|
||||
|
||||
DatagramPacket packet = null;
|
||||
|
||||
packet = new DatagramPacket(
|
||||
DatagramPacket packet = new DatagramPacket(
|
||||
buffer.toByteArray(), buffer.size(),
|
||||
InetAddress.getByName( MDNS_MULTICAST_ADDR ),
|
||||
MDNS_MULTICAST_PORT );
|
||||
|
|
|
|||
|
|
@ -1,48 +1,62 @@
|
|||
package zutil.parser.binary;
|
||||
|
||||
|
||||
import java.io.InvalidClassException;
|
||||
import java.io.InvalidObjectException;
|
||||
import java.lang.reflect.Field;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.*;
|
||||
|
||||
import zutil.ByteUtil;
|
||||
import zutil.converter.Converter;
|
||||
import zutil.parser.binary.BinaryStruct.BinaryField;
|
||||
import zutil.parser.binary.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 int index;
|
||||
private int length;
|
||||
private BinaryFieldSerializer serializer;
|
||||
private Field field;
|
||||
|
||||
|
||||
protected static List<BinaryFieldData> getStructFieldList(Class<? extends BinaryStruct> clazz){
|
||||
if (!cache.containsKey(clazz)) {
|
||||
ArrayList<BinaryFieldData> list = new ArrayList<>();
|
||||
for (Field field : clazz.getDeclaredFields()) {
|
||||
if (field.isAnnotationPresent(BinaryField.class))
|
||||
list.add(new BinaryFieldData(field));
|
||||
try {
|
||||
ArrayList<BinaryFieldData> list = new ArrayList<>();
|
||||
for (Field field : clazz.getDeclaredFields()) {
|
||||
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);
|
||||
}
|
||||
|
||||
|
||||
private BinaryFieldData(Field f){
|
||||
private BinaryFieldData(Field f) throws IllegalAccessException, InstantiationException {
|
||||
field = f;
|
||||
BinaryField fieldData = field.getAnnotation(BinaryField.class);
|
||||
index = fieldData.index();
|
||||
length = fieldData.length();
|
||||
if (field.isAnnotationPresent(CustomBinaryField.class)){
|
||||
CustomBinaryField fieldData = field.getAnnotation(CustomBinaryField.class);
|
||||
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){
|
||||
|
|
@ -54,6 +68,8 @@ public class BinaryFieldData implements Comparable<BinaryFieldData> {
|
|||
field.set(obj, Converter.toInt(data));
|
||||
else if (field.getType() == String.class)
|
||||
field.set(obj, new String(data));
|
||||
else
|
||||
throw new UnsupportedOperationException("Unsupported BinaryStruct field class: "+ field.getClass());
|
||||
} catch (IllegalAccessException e){
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
|
@ -81,8 +97,4 @@ public class BinaryFieldData implements Comparable<BinaryFieldData> {
|
|||
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;
|
||||
|
||||
/**
|
||||
* Created by ezivkoc on 2016-01-28.
|
||||
* Created by Ziver on 2016-01-28.
|
||||
*/
|
||||
public interface BinaryStruct {
|
||||
|
||||
|
|
@ -40,4 +40,11 @@ public interface BinaryStruct {
|
|||
int index();
|
||||
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