1*64b9d721Schristos /* $NetBSD: as.c,v 1.4 2011/05/10 01:52:49 christos Exp $ */
2f5d592cfSatatat
3f5d592cfSatatat /*
4f5d592cfSatatat * Copyright (c) 2001 The NetBSD Foundation, Inc.
5f5d592cfSatatat * All rights reserved.
6f5d592cfSatatat *
7f5d592cfSatatat * This code is derived from software contributed to The NetBSD Foundation
8f5d592cfSatatat * by Andrew Brown.
9f5d592cfSatatat *
10f5d592cfSatatat * Redistribution and use in source and binary forms, with or without
11f5d592cfSatatat * modification, are permitted provided that the following conditions
12f5d592cfSatatat * are met:
13f5d592cfSatatat * 1. Redistributions of source code must retain the above copyright
14f5d592cfSatatat * notice, this list of conditions and the following disclaimer.
15f5d592cfSatatat * 2. Redistributions in binary form must reproduce the above copyright
16f5d592cfSatatat * notice, this list of conditions and the following disclaimer in the
17f5d592cfSatatat * documentation and/or other materials provided with the distribution.
18f5d592cfSatatat *
19f5d592cfSatatat * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20f5d592cfSatatat * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21f5d592cfSatatat * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22f5d592cfSatatat * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23f5d592cfSatatat * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24f5d592cfSatatat * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25f5d592cfSatatat * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26f5d592cfSatatat * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27f5d592cfSatatat * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28f5d592cfSatatat * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29f5d592cfSatatat * POSSIBILITY OF SUCH DAMAGE.
30f5d592cfSatatat */
31f5d592cfSatatat
32f5d592cfSatatat #include <sys/cdefs.h>
33f5d592cfSatatat #include <sys/types.h>
34f5d592cfSatatat #include <sys/socket.h>
35f5d592cfSatatat #include <netinet/in.h>
36f5d592cfSatatat #include <arpa/inet.h>
37f5d592cfSatatat #include <netdb.h>
38f5d592cfSatatat #include <unistd.h>
39f5d592cfSatatat #include <string.h>
40f5d592cfSatatat #include <stdlib.h>
41f5d592cfSatatat #include <errno.h>
42f5d592cfSatatat #include <err.h>
43f5d592cfSatatat #include <stdio.h>
44f5d592cfSatatat
45f5d592cfSatatat #include "as.h"
46f5d592cfSatatat
47f5d592cfSatatat #define DEFAULT_AS_SERVER "whois.radb.net"
48f5d592cfSatatat #undef AS_DEBUG_FILE
49f5d592cfSatatat
50f5d592cfSatatat struct aslookup {
51f5d592cfSatatat FILE *as_f;
52f5d592cfSatatat #ifdef AS_DEBUG_FILE
53f5d592cfSatatat FILE *as_debug;
54f5d592cfSatatat #endif /* AS_DEBUG_FILE */
55f5d592cfSatatat };
56f5d592cfSatatat
57f5d592cfSatatat void *
as_setup(const char * server)58*64b9d721Schristos as_setup(const char *server)
59f5d592cfSatatat {
60f5d592cfSatatat struct aslookup *asn;
61*64b9d721Schristos struct addrinfo hints, *res0, *res;
62f5d592cfSatatat FILE *f;
63*64b9d721Schristos int s, error;
64f5d592cfSatatat
65*64b9d721Schristos s = -1;
66*64b9d721Schristos if (server == NULL)
67*64b9d721Schristos server = getenv("RA_SERVER");
68f5d592cfSatatat if (server == NULL)
69f5d592cfSatatat server = DEFAULT_AS_SERVER;
70f5d592cfSatatat
71*64b9d721Schristos memset(&hints, 0, sizeof(hints));
72*64b9d721Schristos hints.ai_family = PF_UNSPEC;
73*64b9d721Schristos hints.ai_socktype = SOCK_STREAM;
74*64b9d721Schristos error = getaddrinfo(server, "whois", &hints, &res0);
75*64b9d721Schristos if (error == EAI_SERVICE) {
76f5d592cfSatatat warnx("warning: whois/tcp service not found");
77*64b9d721Schristos error = getaddrinfo(server, "43", &hints, &res0);
78*64b9d721Schristos }
79f5d592cfSatatat
80*64b9d721Schristos if (error != 0) {
81*64b9d721Schristos warnx("%s: %s", server, gai_strerror(error));
82f5d592cfSatatat return (NULL);
83f5d592cfSatatat }
84f5d592cfSatatat
85*64b9d721Schristos for (res = res0; res; res = res->ai_next) {
86*64b9d721Schristos s = socket(res->ai_family, res->ai_socktype, res->ai_protocol);
87*64b9d721Schristos if (s < 0)
88*64b9d721Schristos continue;
89*64b9d721Schristos if (connect(s, res->ai_addr, res->ai_addrlen) >= 0)
90f5d592cfSatatat break;
91f5d592cfSatatat close(s);
92f5d592cfSatatat s = -1;
93f5d592cfSatatat }
94*64b9d721Schristos freeaddrinfo(res0);
95*64b9d721Schristos if (s < 0) {
96f5d592cfSatatat warn("connect");
97f5d592cfSatatat return (NULL);
98f5d592cfSatatat }
99f5d592cfSatatat
100f5d592cfSatatat f = fdopen(s, "r+");
101f5d592cfSatatat (void)fprintf(f, "!!\n");
102f5d592cfSatatat (void)fflush(f);
103f5d592cfSatatat
104f5d592cfSatatat asn = malloc(sizeof(struct aslookup));
105f5d592cfSatatat if (asn == NULL)
106f5d592cfSatatat (void)fclose(f);
107f5d592cfSatatat else
108f5d592cfSatatat asn->as_f = f;
109f5d592cfSatatat
110f5d592cfSatatat #ifdef AS_DEBUG_FILE
111ecde3f37Swiz if (asn) {
112f5d592cfSatatat asn->as_debug = fopen(AS_DEBUG_FILE, "w");
113f5d592cfSatatat if (asn->as_debug) {
114f5d592cfSatatat (void)fprintf(asn->as_debug, ">> !!\n");
115f5d592cfSatatat (void)fflush(asn->as_debug);
116f5d592cfSatatat }
117ecde3f37Swiz }
118f5d592cfSatatat #endif /* AS_DEBUG_FILE */
119f5d592cfSatatat
120f5d592cfSatatat return (asn);
121f5d592cfSatatat }
122f5d592cfSatatat
123*64b9d721Schristos unsigned int
as_lookup(void * _asn,char * addr,sa_family_t family)124*64b9d721Schristos as_lookup(void *_asn, char *addr, sa_family_t family)
125f5d592cfSatatat {
126f5d592cfSatatat struct aslookup *asn = _asn;
127f5d592cfSatatat char buf[1024];
128*64b9d721Schristos unsigned int as;
129*64b9d721Schristos int rc, dlen, plen;
130f5d592cfSatatat
131*64b9d721Schristos as = 0;
132*64b9d721Schristos rc = dlen = 0;
133*64b9d721Schristos plen = (family == AF_INET6) ? 128 : 32;
134*64b9d721Schristos (void)fprintf(asn->as_f, "!r%s/%d,l\n", addr, plen);
135f5d592cfSatatat (void)fflush(asn->as_f);
136f5d592cfSatatat
137f5d592cfSatatat #ifdef AS_DEBUG_FILE
138f5d592cfSatatat if (asn->as_debug) {
139*64b9d721Schristos (void)fprintf(asn->as_debug, ">> !r%s/%d,l\n", addr, plen);
140f5d592cfSatatat (void)fflush(asn->as_debug);
141f5d592cfSatatat }
142f5d592cfSatatat #endif /* AS_DEBUG_FILE */
143f5d592cfSatatat
144f5d592cfSatatat while (fgets(buf, sizeof(buf), asn->as_f) != NULL) {
145f5d592cfSatatat buf[sizeof(buf) - 1] = '\0';
146f5d592cfSatatat
147f5d592cfSatatat #ifdef AS_DEBUG_FILE
148f5d592cfSatatat if (asn->as_debug) {
149f5d592cfSatatat (void)fprintf(asn->as_debug, "<< %s", buf);
150f5d592cfSatatat (void)fflush(asn->as_debug);
151f5d592cfSatatat }
152f5d592cfSatatat #endif /* AS_DEBUG_FILE */
153f5d592cfSatatat
154f5d592cfSatatat if (rc == 0) {
155f5d592cfSatatat rc = buf[0];
156f5d592cfSatatat switch (rc) {
157f5d592cfSatatat case 'A':
158f5d592cfSatatat /* A - followed by # bytes of answer */
159f5d592cfSatatat sscanf(buf, "A%d\n", &dlen);
160f5d592cfSatatat #ifdef AS_DEBUG_FILE
161f5d592cfSatatat if (asn->as_debug) {
162f5d592cfSatatat (void)fprintf(asn->as_debug,
163f5d592cfSatatat "dlen: %d\n", dlen);
164f5d592cfSatatat (void)fflush(asn->as_debug);
165f5d592cfSatatat }
166f5d592cfSatatat #endif /* AS_DEBUG_FILE */
167f5d592cfSatatat break;
168f5d592cfSatatat case 'C':
169f5d592cfSatatat case 'D':
170f5d592cfSatatat case 'E':
171f5d592cfSatatat case 'F':
172f5d592cfSatatat /* C - no data returned */
173f5d592cfSatatat /* D - key not found */
174f5d592cfSatatat /* E - multiple copies of key */
175f5d592cfSatatat /* F - some other error */
176f5d592cfSatatat break;
177f5d592cfSatatat }
178f5d592cfSatatat if (rc == 'A')
179f5d592cfSatatat /* skip to next input line */
180f5d592cfSatatat continue;
181f5d592cfSatatat }
182f5d592cfSatatat
183f5d592cfSatatat if (dlen == 0)
184f5d592cfSatatat /* out of data, next char read is end code */
185f5d592cfSatatat rc = buf[0];
186f5d592cfSatatat if (rc != 'A')
187f5d592cfSatatat /* either an error off the bat, or a done code */
188f5d592cfSatatat break;
189f5d592cfSatatat
190f5d592cfSatatat /* data received, thank you */
191f5d592cfSatatat dlen -= strlen(buf);
192f5d592cfSatatat
193f5d592cfSatatat /* origin line is the interesting bit */
194f5d592cfSatatat if (as == 0 && strncasecmp(buf, "origin:", 7) == 0) {
195*64b9d721Schristos sscanf(buf + 7, " AS%u", &as);
196f5d592cfSatatat #ifdef AS_DEBUG_FILE
197f5d592cfSatatat if (asn->as_debug) {
198f5d592cfSatatat (void)fprintf(asn->as_debug, "as: %d\n", as);
199f5d592cfSatatat (void)fflush(asn->as_debug);
200f5d592cfSatatat }
201f5d592cfSatatat #endif /* AS_DEBUG_FILE */
202f5d592cfSatatat }
203f5d592cfSatatat }
204f5d592cfSatatat
205f5d592cfSatatat return (as);
206f5d592cfSatatat }
207f5d592cfSatatat
208f5d592cfSatatat void
as_shutdown(void * _asn)209*64b9d721Schristos as_shutdown(void *_asn)
210f5d592cfSatatat {
211f5d592cfSatatat struct aslookup *asn = _asn;
212f5d592cfSatatat
213f5d592cfSatatat (void)fprintf(asn->as_f, "!q\n");
214f5d592cfSatatat (void)fclose(asn->as_f);
215f5d592cfSatatat
216f5d592cfSatatat #ifdef AS_DEBUG_FILE
217f5d592cfSatatat if (asn->as_debug) {
218f5d592cfSatatat (void)fprintf(asn->as_debug, ">> !q\n");
219f5d592cfSatatat (void)fclose(asn->as_debug);
220f5d592cfSatatat }
221f5d592cfSatatat #endif /* AS_DEBUG_FILE */
222f5d592cfSatatat
223f5d592cfSatatat free(asn);
224f5d592cfSatatat }
225