-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdns_server.ts
134 lines (123 loc) · 5.79 KB
/
dns_server.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
import { DNSPacket, DNSQuestion } from "./dns_packet.ts";
import { DNSConfigRecord, DNSConfig } from "./dns_server_config.ts";
import { DNSRecordClass } from "./dns_record_class.ts";
import { DNSRecordType, AResourceRecord, ResourceRecord, CNameResourceRecord, AAAAResourceRecord } from "./dns_record_type.ts";
import { ipv4ToNumber, ipv6ToBytes } from "./utils.ts";
/** A simple DNS Server. */
export class DNSServer {
/**
* Creates a new DNSServer.
*
* @param records The records that should be served by this server.
*/
constructor(private readonly records:DNSConfig){}
/**
* Handles a raw DNS request. Request payload should be the raw datagram
* content.
*
* Returns the raw bytes for a DNS response to the request.
*/
public HandleRequest(request: Uint8Array): Uint8Array {
const packet = DNSPacket.fromBytes(request);
const header = packet.Header;
const question = packet.Question;
let records:DNSConfig[] = [];
try {
// Special handling for A records: if we don't have an A record, check to
// see if we have a CNAME for it, then get the A record for the CNAME
// destination if we do.
//
// This is special processing only for CNAMEs - see the RFC for details:
// https://tools.ietf.org/html/rfc1034#section-3.6.2
if (question.RecordType == DNSRecordType.A ||
question.RecordType == DNSRecordType.AAAA) {
// This was an A/AAAA record request and we *don't* have an A/AAA record
// then handle the CNAME special case.
if (!this.hasRecord(question.Name, question.RecordType)) {
// No A/AAAA record found for this name - look for a CNAME instead.
let cnameRecord = this.getRecord(question.Name, DNSRecordType.CNAME);
if (cnameRecord) {
// We have a CNAME - add it to the response and then see if we have
// an A/AAAA for the CNAME's destination.
records.push(cnameRecord);
const key = Object.keys(cnameRecord)[0]; // This feels wrong?
const cnameDestination = cnameRecord[key].class[DNSRecordClass[question.RecordClass]][DNSRecordType[DNSRecordType.CNAME]];
if (this.hasRecord(cnameDestination, question.RecordType)) {
// Yes we have a A/AAAA for this CNAME dest - add it to response.
records.push(this.getRecord(cnameDestination, question.RecordType));
}
}
} else {
// A/AAAA was found - just add that to the response.
records.push(this.getRecord(question.Name, question.RecordType));
}
} else {
// Not an A record request - carry on as usual.
records.push(this.getRecord(question.Name, question.RecordType));
}
} catch (error) {
console.error(`Error handling request: ${error}`);
return request;
}
console.log(`Serving request: ${packet.Question}`);
packet.Header.Flags = 32768; // 0x8000
for (const record of records) {
const rrType = this.getResourceRecordType(packet.Question, record);
if (rrType) packet.Answers.push(rrType);
}
packet.Header.TotalAnswers = packet.Answers.length;
return new Uint8Array(packet.Bytes);
}
/**
* Checks for a config record by name, type, and optionally class (class
* defaults to `IN` if not set).
*/
private hasRecord(name:string,
recordType:DNSRecordType,
recordClass:DNSRecordClass = DNSRecordClass.IN): boolean {
const config = this.records[name];
if (!config) return false;
if (!config.class[DNSRecordClass[recordClass]]) return false;
if (!config.class[DNSRecordClass[recordClass]][DNSRecordType[recordType]]) return false;
return true;
}
/**
* Get the config record by name, type, and optionally class (class defaults
* to `IN` if not set).
*/
private getRecord(name:string,
recordType:DNSRecordType,
recordClass:DNSRecordClass = DNSRecordClass.IN): DNSConfig {
const config = this.records[name];
if (!config) throw new Error(`No config for ${name}`);
if (!config.class[DNSRecordClass[recordClass]]) throw new Error(`No config for class '${recordClass}' for ${name}`);
if (!config.class[DNSRecordClass[recordClass]][DNSRecordType[recordType]]) throw new Error(`No config for type '${recordType}' for ${name}`);
return {[name]: config};
}
/** Get an appropriate record type for the question using the config. */
// TODO: refactor this whole thing - should be per-record, not relating to Question.
// TODO: allow both A & AAAA to be returned at once.
private getResourceRecordType(question:DNSQuestion, config:DNSConfig):ResourceRecord | undefined {
const key = Object.keys(config)[0]; // This feels wrong?
const classConfig = config[key].class[DNSRecordClass[question.RecordClass]];
let rr:ResourceRecord|undefined;
// TODO: make records strongly typed to avoid this mess
if (classConfig.hasOwnProperty(DNSRecordType[DNSRecordType.A])) {
rr = new AResourceRecord(key, key.split('.'),
question.RecordType, question.RecordClass, config[key].ttl);
(rr as AResourceRecord).Address =
ipv4ToNumber(classConfig[DNSRecordType[DNSRecordType.A]]);
} else if (classConfig.hasOwnProperty(DNSRecordType[DNSRecordType.AAAA])) {
rr = new AAAAResourceRecord(key, key.split('.'),
question.RecordType, question.RecordClass, config[key].ttl);
(rr as AAAAResourceRecord).Address =
ipv6ToBytes(classConfig[DNSRecordType[DNSRecordType.AAAA]]);
} else if (classConfig.hasOwnProperty(DNSRecordType[DNSRecordType.CNAME])) {
const name = classConfig[DNSRecordType[DNSRecordType.CNAME]];
rr = new CNameResourceRecord(key, key.split('.'),
DNSRecordType.CNAME, question.RecordClass, config[key].ttl);
(rr as CNameResourceRecord).CName = name;
}
return rr;
}
}