blob: acb3e3bb2718dd58dde41c688dda785d2eb3359a [file] [log] [blame]
Erik Andersene49d5ec2000-02-08 19:58:47 +00001/* vi: set sw=4 ts=4: */
John Beppub332e772000-01-29 12:59:01 +00002/*
3 * Mini nslookup implementation for busybox
4 *
John Beppub332e772000-01-29 12:59:01 +00005 * Copyright (C) 2000 by Lineo, inc.
6 * Written by John Beppu <beppu@lineo.com>
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 *
22 */
23
Eric Andersen3570a342000-09-25 21:45:58 +000024#include "busybox.h"
John Beppub332e772000-01-29 12:59:01 +000025#include <ctype.h>
26#include <errno.h>
27#include <stdio.h>
28#include <string.h>
29
30#include <netdb.h>
31#include <sys/socket.h>
32#include <sys/types.h>
33#include <netinet/in.h>
34
John Beppu50bc1012000-01-30 09:47:16 +000035/*
36 | I'm only implementing non-interactive mode;
37 | I totally forgot nslookup even had an interactive mode.
38 |
39 | [ TODO ]
40 | + find out how to use non-default name servers
41 | + find out how the real nslookup gets the default name server
42 */
John Beppub332e772000-01-29 12:59:01 +000043
John Beppu50bc1012000-01-30 09:47:16 +000044/* I have to see how the real nslookup does this.
45 * I could dig through /etc/resolv.conf, but is there a
46 * better (programatic) way?
47 */
Erik Andersene49d5ec2000-02-08 19:58:47 +000048static void server_fprint(FILE * dst)
John Beppub332e772000-01-29 12:59:01 +000049{
Erik Andersen5e1189e2000-04-15 16:34:54 +000050 fprintf(dst, "Server: %s\n", "default");
51 fprintf(dst, "Address: %s\n\n", "default");
John Beppub332e772000-01-29 12:59:01 +000052}
53
54/* only works for IPv4 */
Erik Andersene49d5ec2000-02-08 19:58:47 +000055static int addr_fprint(char *addr, FILE * dst)
John Beppub332e772000-01-29 12:59:01 +000056{
Erik Andersene2729152000-02-18 21:34:17 +000057 u_int8_t split[4];
58 u_int32_t ip;
59 u_int32_t *x = (u_int32_t *) addr;
John Beppub332e772000-01-29 12:59:01 +000060
Erik Andersene49d5ec2000-02-08 19:58:47 +000061 ip = ntohl(*x);
62 split[0] = (ip & 0xff000000) >> 24;
63 split[1] = (ip & 0x00ff0000) >> 16;
64 split[2] = (ip & 0x0000ff00) >> 8;
65 split[3] = (ip & 0x000000ff);
66 fprintf(dst, "%d.%d.%d.%d", split[0], split[1], split[2], split[3]
67 );
68 return 0;
John Beppub332e772000-01-29 12:59:01 +000069}
70
John Beppu50bc1012000-01-30 09:47:16 +000071/* changes a c-string matching the perl regex \d+\.\d+\.\d+\.\d+
Erik Andersene2729152000-02-18 21:34:17 +000072 * into a u_int32_t
John Beppu50bc1012000-01-30 09:47:16 +000073 */
Erik Andersene2729152000-02-18 21:34:17 +000074static u_int32_t str_to_addr(const char *addr)
John Beppub332e772000-01-29 12:59:01 +000075{
Erik Andersene2729152000-02-18 21:34:17 +000076 u_int32_t split[4];
77 u_int32_t ip;
John Beppub332e772000-01-29 12:59:01 +000078
Erik Andersene49d5ec2000-02-08 19:58:47 +000079 sscanf(addr, "%d.%d.%d.%d",
80 &split[0], &split[1], &split[2], &split[3]);
John Beppub332e772000-01-29 12:59:01 +000081
Erik Andersene49d5ec2000-02-08 19:58:47 +000082 /* assuming sscanf worked */
83 ip = (split[0] << 24) |
84 (split[1] << 16) | (split[2] << 8) | (split[3]);
John Beppub332e772000-01-29 12:59:01 +000085
Erik Andersene49d5ec2000-02-08 19:58:47 +000086 return htonl(ip);
John Beppub332e772000-01-29 12:59:01 +000087}
88
John Beppu50bc1012000-01-30 09:47:16 +000089/* takes the NULL-terminated array h_addr_list, and
90 * prints its contents appropriately
91 */
Erik Andersene49d5ec2000-02-08 19:58:47 +000092static int addr_list_fprint(char **h_addr_list, FILE * dst)
John Beppub332e772000-01-29 12:59:01 +000093{
Erik Andersene49d5ec2000-02-08 19:58:47 +000094 int i, j;
95 char *addr_string = (h_addr_list[1])
Erik Andersen5e1189e2000-04-15 16:34:54 +000096 ? "Addresses: " : "Address: ";
John Beppub332e772000-01-29 12:59:01 +000097
Erik Andersen5e1189e2000-04-15 16:34:54 +000098 fprintf(dst, "%s ", addr_string);
Erik Andersene49d5ec2000-02-08 19:58:47 +000099 for (i = 0, j = 0; h_addr_list[i]; i++, j++) {
100 addr_fprint(h_addr_list[i], dst);
John Beppu50bc1012000-01-30 09:47:16 +0000101
Erik Andersene49d5ec2000-02-08 19:58:47 +0000102 /* real nslookup does this */
103 if (j == 4) {
104 if (h_addr_list[i + 1]) {
105 fprintf(dst, "\n ");
106 }
107 j = 0;
108 } else {
109 if (h_addr_list[i + 1]) {
110 fprintf(dst, ", ");
111 }
112 }
113
John Beppub332e772000-01-29 12:59:01 +0000114 }
Erik Andersene49d5ec2000-02-08 19:58:47 +0000115 fprintf(dst, "\n");
116 return 0;
John Beppub332e772000-01-29 12:59:01 +0000117}
118
John Beppu50bc1012000-01-30 09:47:16 +0000119/* gethostbyaddr wrapper */
Erik Andersene49d5ec2000-02-08 19:58:47 +0000120static struct hostent *gethostbyaddr_wrapper(const char *address)
John Beppub332e772000-01-29 12:59:01 +0000121{
Erik Andersene49d5ec2000-02-08 19:58:47 +0000122 struct in_addr addr;
John Beppub332e772000-01-29 12:59:01 +0000123
Erik Andersene49d5ec2000-02-08 19:58:47 +0000124 addr.s_addr = str_to_addr(address);
125 return gethostbyaddr((char *) &addr, 4, AF_INET); /* IPv4 only for now */
John Beppu50bc1012000-01-30 09:47:16 +0000126}
127
128/* print the results as nslookup would */
Erik Andersene49d5ec2000-02-08 19:58:47 +0000129static struct hostent *hostent_fprint(struct hostent *host, FILE * dst)
John Beppu50bc1012000-01-30 09:47:16 +0000130{
Erik Andersene49d5ec2000-02-08 19:58:47 +0000131 if (host) {
Erik Andersen5e1189e2000-04-15 16:34:54 +0000132 fprintf(dst, "Name: %s\n", host->h_name);
Erik Andersene49d5ec2000-02-08 19:58:47 +0000133 addr_list_fprint(host->h_addr_list, dst);
134 } else {
Erik Andersen5afc8642000-05-02 00:07:56 +0000135 fprintf(dst, "*** Unknown host\n");
Erik Andersene49d5ec2000-02-08 19:58:47 +0000136 }
137 return host;
John Beppub332e772000-01-29 12:59:01 +0000138}
139
John Beppub332e772000-01-29 12:59:01 +0000140
John Beppu50bc1012000-01-30 09:47:16 +0000141/* naive function to check whether char *s is an ip address */
Erik Andersene49d5ec2000-02-08 19:58:47 +0000142static int is_ip_address(const char *s)
John Beppub332e772000-01-29 12:59:01 +0000143{
Erik Andersene49d5ec2000-02-08 19:58:47 +0000144 while (*s) {
145 if ((isdigit(*s)) || (*s == '.')) {
146 s++;
147 continue;
148 }
149 return 0;
150 }
151 return 1;
John Beppub332e772000-01-29 12:59:01 +0000152}
153
154/* ________________________________________________________________________ */
Erik Andersene49d5ec2000-02-08 19:58:47 +0000155int nslookup_main(int argc, char **argv)
John Beppub332e772000-01-29 12:59:01 +0000156{
Erik Andersene49d5ec2000-02-08 19:58:47 +0000157 struct hostent *host;
John Beppu50bc1012000-01-30 09:47:16 +0000158
Erik Andersen5e1189e2000-04-15 16:34:54 +0000159 if (argc < 2 || *argv[1]=='-') {
Erik Andersene49d5ec2000-02-08 19:58:47 +0000160 usage(nslookup_usage);
161 }
John Beppub332e772000-01-29 12:59:01 +0000162
Erik Andersene49d5ec2000-02-08 19:58:47 +0000163 server_fprint(stdout);
164 if (is_ip_address(argv[1])) {
165 host = gethostbyaddr_wrapper(argv[1]);
166 } else {
167 host = gethostbyname(argv[1]);
168 }
169 hostent_fprint(host, stdout);
Eric Andersenb6106152000-06-19 17:25:40 +0000170 return( TRUE);
John Beppub332e772000-01-29 12:59:01 +0000171}
172
Eric Andersen3570a342000-09-25 21:45:58 +0000173/* $Id: nslookup.c,v 1.12 2000/09/25 21:45:58 andersen Exp $ */