forked from mageddo/dns-proxy-server
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathapp.js
More file actions
97 lines (82 loc) · 2.91 KB
/
app.js
File metadata and controls
97 lines (82 loc) · 2.91 KB
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
'use strict';
let recordsManager = require('./recordsManager/index.js')
let dns = require('native-dns');
let server = dns.createServer();
let async = require('async');
let qtypeToName = require('native-dns-packet').consts.qtypeToName;
server.on('listening', () => console.log('server listening on', server.address()));
server.on('close', () => console.log('server closed', server.address()));
server.on('error', (err, buff, req, res) => console.error(err.stack));
server.on('socketError', (err, socket) => console.error(err));
server.serve(53);
function proxy(question, response, cb) {
console.log('proxying: ', question.name, ', type: ', question.type);
let server = recordsManager.data.remoteDns[0];
if(!server){
throw "You need at least one remote server";
}
let request = dns.Request({
question: question, // forwarding the question
server: server, // this is the DNS server we are asking
timeout: 1000
});
request.on('timeout', function () {
console.log('Timeout in making request no forwarding', question.name);
});
// when we get answers, append them to the response
request.on('message', (err, msg) => {
msg.answer.forEach(a => {
response.answer.push(a);
});
msg.authority.forEach(a => {
response.answer.push(a);
});
});
request.on('end', function(){
response.answer.forEach(msg => {
console.log('remote DNS answer: type: ', msg.type, ', name: ', msg.name, ', address: ', msg.address);
})
cb();
});
request.send();
}
server.on('request', function handleRequest(request, response) {
let questionsToProxy = [];
/**
* DNS server can receive many questions on a same request,
* so we needle 'ask' all requests locally or remotelly then
* respond to the reponse variable with the found results calling response.send()
*
*/
var nQuestions = request.question.length;
console.log('request with ', nQuestions, ' questions');
request.question.forEach(question => {
console.log('request from:', request.address.address, ' for:', question.name, ' type:', qtypeToName(question.type));
// finding a entry on local base that matches with question
let entry = recordsManager.data.entries.filter(r => new RegExp(r.domain, 'i').exec(question.name));
if (entry.length) {
entry[0].records.forEach(record => {
record.name = question.name;
record.ttl = record.ttl || 1800;
if (record.type == 'CNAME') {
record.data = record.address;
questionsToProxy.push(cb => {
proxy({
name: record.data, type: dns.consts.NAME_TO_QTYPE.A, class: 1
}, response, cb);
});
}
response.answer.push(dns[record.type](record));
});
} else {
// forwarding host
questionsToProxy.push(cb => proxy(question, response, cb));
}
});
async.parallel(questionsToProxy, function() {
// when all questions be done (end event) we will close the connection
// sending the response
console.log('all questions be done, send client answers');
response.send();
});
});