-
Notifications
You must be signed in to change notification settings - Fork 11
/
oas112d.c
115 lines (99 loc) · 4.08 KB
/
oas112d.c
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
/*
* $Id: as112d.c 29 2010-01-15 11:54:55Z ray.bellis $
*
* Copyright (c) 2009, Nominet UK.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of Nominet UK nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY Nominet UK ''AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL Nominet UK BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include <stdlib.h>
#include <stdio.h>
#include <ctype.h>
#include <evldns.h>
static char *t_soa = "@ SOA a.as112.net. hostmaster.as112.net. 1 604800 2592000 0604800 604800";
static char *t_ns1 = "@ NS b.as112.net.";
static char *t_ns2 = "@ NS c.as112.net.";
/* rejects packets that arrive with OPCODE != QUERY, or QDCOUNT != 1 */
void query_only(evldns_server_request *srq, void *user_data, ldns_rdf *qname, ldns_rr_type qtype, ldns_rr_class qclass)
{
ldns_pkt *req = srq->request;
if (ldns_pkt_get_opcode(req) != LDNS_PACKET_QUERY) {
srq->response = evldns_response(req, LDNS_RCODE_NOTIMPL);
}
if (ldns_pkt_qdcount(req) != 1) {
srq->response = evldns_response(req, LDNS_RCODE_FORMERR);
}
}
void as112_callback(evldns_server_request *srq, void *user_data, ldns_rdf *qname, ldns_rr_type qtype, ldns_rr_class qclass)
{
/* the default response packet */
ldns_pkt *resp = srq->response = evldns_response(srq->request, LDNS_RCODE_NOERROR);
/* misc local variables */
ldns_rr_list *answer, *authority;
ldns_rr *soa, *ns1, *ns2;
int ancount = 0;
/* we do not support zone transfers */
if (qtype == LDNS_RR_TYPE_AXFR || qtype == LDNS_RR_TYPE_IXFR) {
ldns_pkt_set_rcode(resp, LDNS_RCODE_NOTIMPL);
return;
}
/* we need references to the response's sections */
answer = ldns_pkt_answer(resp);
authority = ldns_pkt_authority(resp);
/* SOA */
if (qtype == LDNS_RR_TYPE_ANY || qtype == LDNS_RR_TYPE_SOA) {
ldns_rr_new_frm_str(&soa, t_soa, 300, qname, NULL);
ldns_rr_list_push_rr(answer, soa);
}
/* NS */
if (qtype == LDNS_RR_TYPE_ANY || qtype == LDNS_RR_TYPE_NS) {
ldns_rr_new_frm_str(&ns1, t_ns1, 300, qname, NULL);
ldns_rr_new_frm_str(&ns2, t_ns2, 300, qname, NULL);
ldns_rr_list_push_rr(answer, ns1);
ldns_rr_list_push_rr(answer, ns2);
}
/* if NODATA fill authority section with an SOA */
ancount = ldns_rr_list_rr_count(answer);
ldns_pkt_set_ancount(resp, ancount);
if (!ancount) {
ldns_rr_new_frm_str(&soa, t_soa, 300, qname, NULL);
ldns_rr_list_push_rr(authority, soa);
ldns_pkt_set_nscount(resp, 1);
}
/* update packet header */
ldns_pkt_set_aa(resp, 1);
}
int main(int argc, char *argv[])
{
struct evldns_server *p;
struct event_base *base;
base = event_base_new();
p = evldns_add_server(base);
evldns_add_server_port(p, bind_to_udp4_port(5053));
evldns_add_server_port(p, bind_to_tcp4_port(5053, 10));
evldns_add_callback(p, NULL, LDNS_RR_CLASS_ANY, LDNS_RR_TYPE_ANY, query_only, NULL);
evldns_add_callback(p, NULL, LDNS_RR_CLASS_ANY, LDNS_RR_TYPE_ANY, as112_callback, NULL);
event_base_dispatch(base);
return EXIT_SUCCESS;
}