it works minus SRV
This commit is contained in:
146
core/src/mindustry/net/Dns.java
Normal file
146
core/src/mindustry/net/Dns.java
Normal file
@@ -0,0 +1,146 @@
|
||||
package mindustry.net;
|
||||
|
||||
import arc.func.*;
|
||||
import arc.math.*;
|
||||
import arc.struct.*;
|
||||
import arc.util.*;
|
||||
|
||||
import java.net.*;
|
||||
import java.nio.*;
|
||||
import java.nio.channels.*;
|
||||
import java.nio.charset.*;
|
||||
import java.util.regex.*;
|
||||
|
||||
public class Dns{
|
||||
private static final Pattern ipPattern = Pattern.compile("^((0|1\\d?\\d?|2[0-4]?\\d?|25[0-5]?|[3-9]\\d?)\\.){3}(0|1\\d?\\d?|2[0-4]?\\d?|25[0-5]?|[3-9]\\d?)$");
|
||||
private static final int aRecord = 1, srvRecord = 33;
|
||||
private static IntMap<ObjectMap<String, Seq<?>>> cache = new IntMap<>();
|
||||
private static Seq<InetSocketAddress> nameservers = Seq.with(new InetSocketAddress("1.1.1.1", 53), new InetSocketAddress("8.8.8.8", 53));
|
||||
|
||||
static <T> void resolve(int type, String domain, Func<ByteBuffer, T> reader, Cons<Seq<T>> result, Cons<Exception> error){
|
||||
ObjectMap<String, Seq<?>> map;
|
||||
|
||||
synchronized(cache){
|
||||
map = cache.get(type, ObjectMap::new);
|
||||
|
||||
//TODO timeout
|
||||
if(map.containsKey(domain)){
|
||||
result.get((Seq<T>)map.get(domain));
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
send(Seq.with(new InetSocketAddress("1.1.1.1", 53)), 0, type, domain, reader, records -> {
|
||||
synchronized(cache){
|
||||
//cache the records
|
||||
map.put(domain, records);
|
||||
}
|
||||
|
||||
result.get(records);
|
||||
}, error);
|
||||
}
|
||||
|
||||
static void resolveAddress(String domain, Cons<InetAddress> result, Cons<Exception> error){
|
||||
if(ipPattern.matcher(domain).matches()){
|
||||
try{
|
||||
result.get(InetAddress.getByName(domain));
|
||||
}catch(Exception e){
|
||||
error.get(e);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
resolve(aRecord, domain, bytes -> {
|
||||
byte[] address = new byte[4];
|
||||
bytes.get(address);
|
||||
return address;
|
||||
}, addresses -> {
|
||||
try{
|
||||
if(addresses.size > 0){
|
||||
result.get(InetAddress.getByAddress(addresses.get(0)));
|
||||
}else{
|
||||
//there are no records found
|
||||
error.get(new UnresolvedAddressException());
|
||||
}
|
||||
}catch(UnknownHostException unknown){
|
||||
error.get(unknown);
|
||||
}
|
||||
}, error);
|
||||
}
|
||||
|
||||
static <T> void send(Seq<InetSocketAddress> addresses, int index, int type, String domain, Func<ByteBuffer, T> reader, Cons<Seq<T>> recordResult, Cons<Exception> error){
|
||||
short id = (short)new Rand().nextInt(Short.MAX_VALUE);
|
||||
|
||||
ByteBuffer buffer = ByteBuffer.allocate(512);
|
||||
|
||||
buffer.putShort(id); // Id
|
||||
buffer.putShort((short) 0x0100); // Flags (recursion enabled)
|
||||
buffer.putShort((short) 1); // Questions
|
||||
buffer.putShort((short) 0); // Answers
|
||||
buffer.putShort((short) 0); // Authority
|
||||
buffer.putShort((short) 0); // Additional
|
||||
|
||||
// Domain
|
||||
for(String part : domain.split("\\.")) {
|
||||
buffer.put((byte) part.length());
|
||||
buffer.put(part.getBytes(StandardCharsets.UTF_8));
|
||||
}
|
||||
buffer.put((byte) 0);
|
||||
|
||||
buffer.putShort((short) type); // Type
|
||||
buffer.putShort((short) 1); // Class (Internet)
|
||||
|
||||
buffer.flip();
|
||||
|
||||
AsyncUdp.send(addresses.get(index), 2000, 512, buffer, result -> {
|
||||
short responseId = result.getShort();
|
||||
if(responseId != id) {
|
||||
throw new ArcRuntimeException("Invalid response ID");
|
||||
}
|
||||
|
||||
result.getShort();
|
||||
result.getShort();
|
||||
int answers = result.getShort() & 0xFFFF;
|
||||
result.getShort();
|
||||
result.getShort();
|
||||
|
||||
byte len;
|
||||
while((len = result.get()) != 0) {
|
||||
result.position(result.position() + len);
|
||||
}
|
||||
|
||||
result.getShort();
|
||||
result.getShort();
|
||||
|
||||
var records = new Seq<T>(answers);
|
||||
|
||||
for(int i = 0; i < answers; i++) {
|
||||
result.getShort(); // OFFSET
|
||||
int answerType = result.getShort() & 0xFFFF; // Type
|
||||
result.getShort(); // Class
|
||||
long ttl = result.getInt() & 0xFFFFFFFFL; // TTL
|
||||
int length = result.getShort() & 0xFFFF; // Data length
|
||||
|
||||
// Optionally CNAME results will be returned with the A results, skip those
|
||||
if(answerType != type){
|
||||
result.position(result.position() + length);
|
||||
continue;
|
||||
}
|
||||
|
||||
int position = result.position();
|
||||
|
||||
records.add(reader.get(result));
|
||||
|
||||
result.position(position + length);
|
||||
}
|
||||
|
||||
recordResult.get(records);
|
||||
}, e -> {
|
||||
if(index >= addresses.size - 1){
|
||||
error.get(e);
|
||||
}else{
|
||||
send(addresses, index + 1, type, domain, reader, recordResult, error);
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user