diff options
Diffstat (limited to 'plugins/netutils.c')
-rw-r--r-- | plugins/netutils.c | 431 |
1 files changed, 431 insertions, 0 deletions
diff --git a/plugins/netutils.c b/plugins/netutils.c new file mode 100644 index 00000000..e5d35281 --- /dev/null +++ b/plugins/netutils.c | |||
@@ -0,0 +1,431 @@ | |||
1 | /**************************************************************************** | ||
2 | * | ||
3 | * Nagios plugins network utilities | ||
4 | * | ||
5 | * License: GPL | ||
6 | * Copyright (c) 1999 Ethan Galstad (nagios@nagios.org) | ||
7 | * | ||
8 | * Last Modified: $Date$ | ||
9 | * | ||
10 | * Description: | ||
11 | * | ||
12 | * This file contains commons functions used in many of the plugins. | ||
13 | * | ||
14 | * License Information: | ||
15 | * | ||
16 | * This program is free software; you can redistribute it and/or modify | ||
17 | * it under the terms of the GNU General Public License as published by | ||
18 | * the Free Software Foundation; either version 2 of the License, or | ||
19 | * (at your option) any later version. | ||
20 | * | ||
21 | * This program is distributed in the hope that it will be useful, | ||
22 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
23 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
24 | * GNU General Public License for more details. | ||
25 | * | ||
26 | * You should have received a copy of the GNU General Public License | ||
27 | * along with this program; if not, write to the Free Software | ||
28 | * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||
29 | * | ||
30 | ****************************************************************************/ | ||
31 | |||
32 | #include "config.h" | ||
33 | #include "common.h" | ||
34 | #include <sys/socket.h> | ||
35 | #include <netinet/in.h> | ||
36 | #include <arpa/inet.h> | ||
37 | #include <netdb.h> | ||
38 | |||
39 | extern int socket_timeout; | ||
40 | RETSIGTYPE socket_timeout_alarm_handler (int); | ||
41 | |||
42 | int process_tcp_request2 (char *, int, char *, char *, int); | ||
43 | int process_tcp_request (char *, int, char *, char *, int); | ||
44 | int process_udp_request (char *, int, char *, char *, int); | ||
45 | int process_request (char *, int, char *, char *, char *, int); | ||
46 | |||
47 | int my_tcp_connect (char *, int, int *); | ||
48 | int my_udp_connect (char *, int, int *); | ||
49 | int my_connect (char *, int, int *, char *); | ||
50 | |||
51 | int my_inet_aton (register const char *, struct in_addr *); | ||
52 | |||
53 | /* handles socket timeouts */ | ||
54 | void | ||
55 | socket_timeout_alarm_handler (int sig) | ||
56 | { | ||
57 | |||
58 | printf ("Socket timeout after %d seconds\n", socket_timeout); | ||
59 | |||
60 | exit (STATE_CRITICAL); | ||
61 | } | ||
62 | |||
63 | |||
64 | /* connects to a host on a specified TCP port, sends a string, | ||
65 | and gets a response */ | ||
66 | int | ||
67 | process_tcp_request (char *server_address, | ||
68 | int server_port, | ||
69 | char *send_buffer, char *recv_buffer, int recv_size) | ||
70 | { | ||
71 | int result; | ||
72 | char proto[4] = "tcp"; | ||
73 | |||
74 | result = process_request (server_address, | ||
75 | server_port, | ||
76 | proto, send_buffer, recv_buffer, recv_size); | ||
77 | |||
78 | return result; | ||
79 | } | ||
80 | |||
81 | |||
82 | /* connects to a host on a specified UDP port, sends a string, and gets a | ||
83 | response */ | ||
84 | int | ||
85 | process_udp_request (char *server_address, | ||
86 | int server_port, | ||
87 | char *send_buffer, char *recv_buffer, int recv_size) | ||
88 | { | ||
89 | int result; | ||
90 | char proto[4] = "udp"; | ||
91 | |||
92 | result = process_request (server_address, | ||
93 | server_port, | ||
94 | proto, send_buffer, recv_buffer, recv_size); | ||
95 | |||
96 | return result; | ||
97 | } | ||
98 | |||
99 | |||
100 | |||
101 | /* connects to a host on a specified tcp port, sends a string, and gets a | ||
102 | response. loops on select-recv until timeout or eof to get all of a | ||
103 | multi-packet answer */ | ||
104 | int | ||
105 | process_tcp_request2 (char *server_address, | ||
106 | int server_port, | ||
107 | char *send_buffer, char *recv_buffer, int recv_size) | ||
108 | { | ||
109 | |||
110 | int result; | ||
111 | int send_result; | ||
112 | int recv_result; | ||
113 | int sd; | ||
114 | struct timeval tv; | ||
115 | fd_set readfds; | ||
116 | int recv_length = 0; | ||
117 | |||
118 | result = my_connect (server_address, server_port, &sd, "tcp"); | ||
119 | if (result != STATE_OK) | ||
120 | return STATE_CRITICAL; | ||
121 | |||
122 | send_result = send (sd, send_buffer, strlen (send_buffer), 0); | ||
123 | if (send_result != strlen (send_buffer)) { | ||
124 | printf ("send() failed\n"); | ||
125 | result = STATE_WARNING; | ||
126 | } | ||
127 | |||
128 | while (1) { | ||
129 | /* wait up to the number of seconds for socket timeout | ||
130 | minus one for data from the host */ | ||
131 | tv.tv_sec = socket_timeout - 1; | ||
132 | tv.tv_usec = 0; | ||
133 | FD_ZERO (&readfds); | ||
134 | FD_SET (sd, &readfds); | ||
135 | select (sd + 1, &readfds, NULL, NULL, &tv); | ||
136 | |||
137 | /* make sure some data has arrived */ | ||
138 | if (!FD_ISSET (sd, &readfds)) { /* it hasn't */ | ||
139 | if (!recv_length) { | ||
140 | strcpy (recv_buffer, ""); | ||
141 | printf ("No data was recieved from host!\n"); | ||
142 | result = STATE_WARNING; | ||
143 | } | ||
144 | else { /* this one failed, but previous ones worked */ | ||
145 | recv_buffer[recv_length] = 0; | ||
146 | } | ||
147 | break; | ||
148 | } | ||
149 | else { /* it has */ | ||
150 | recv_result = | ||
151 | recv (sd, recv_buffer + recv_length, recv_size - recv_length - 1, 0); | ||
152 | if (recv_result == -1) { /* recv failed, bail out */ | ||
153 | strcpy (recv_buffer + recv_length, ""); | ||
154 | result = STATE_WARNING; | ||
155 | break; | ||
156 | } | ||
157 | else if (recv_result == 0) { /* end of file ? */ | ||
158 | recv_buffer[recv_length] = 0; | ||
159 | break; | ||
160 | } | ||
161 | else { /* we got data! */ | ||
162 | recv_length += recv_result; | ||
163 | if (recv_length >= recv_size - 1) { /* buffer full, we're done */ | ||
164 | recv_buffer[recv_size - 1] = 0; | ||
165 | break; | ||
166 | } | ||
167 | } | ||
168 | } /* end if(!FD_ISSET(sd,&readfds)) */ | ||
169 | } /* end while(1) */ | ||
170 | |||
171 | close (sd); | ||
172 | return result; | ||
173 | } | ||
174 | |||
175 | /* connects to a host on a specified port, sends a string, and gets a | ||
176 | response */ | ||
177 | int | ||
178 | process_request (char *server_address, | ||
179 | int server_port, | ||
180 | char *proto, | ||
181 | char *send_buffer, char *recv_buffer, int recv_size) | ||
182 | { | ||
183 | int result; | ||
184 | int send_result; | ||
185 | int recv_result; | ||
186 | int sd; | ||
187 | struct timeval tv; | ||
188 | fd_set readfds; | ||
189 | |||
190 | result = STATE_OK; | ||
191 | |||
192 | result = my_connect (server_address, server_port, &sd, proto); | ||
193 | if (result != STATE_OK) | ||
194 | return STATE_CRITICAL; | ||
195 | |||
196 | send_result = send (sd, send_buffer, strlen (send_buffer), 0); | ||
197 | if (send_result != strlen (send_buffer)) { | ||
198 | printf ("send() failed\n"); | ||
199 | result = STATE_WARNING; | ||
200 | } | ||
201 | |||
202 | /* wait up to the number of seconds for socket timeout minus one | ||
203 | for data from the host */ | ||
204 | tv.tv_sec = socket_timeout - 1; | ||
205 | tv.tv_usec = 0; | ||
206 | FD_ZERO (&readfds); | ||
207 | FD_SET (sd, &readfds); | ||
208 | select (sd + 1, &readfds, NULL, NULL, &tv); | ||
209 | |||
210 | /* make sure some data has arrived */ | ||
211 | if (!FD_ISSET (sd, &readfds)) { | ||
212 | strcpy (recv_buffer, ""); | ||
213 | printf ("No data was recieved from host!\n"); | ||
214 | result = STATE_WARNING; | ||
215 | } | ||
216 | |||
217 | else { | ||
218 | recv_result = recv (sd, recv_buffer, recv_size - 1, 0); | ||
219 | if (recv_result == -1) { | ||
220 | strcpy (recv_buffer, ""); | ||
221 | if (!strcmp (proto, "tcp")) | ||
222 | printf ("recv() failed\n"); | ||
223 | result = STATE_WARNING; | ||
224 | } | ||
225 | else | ||
226 | recv_buffer[recv_result] = 0; | ||
227 | |||
228 | /* terminate returned string */ | ||
229 | recv_buffer[recv_size - 1] = 0; | ||
230 | } | ||
231 | |||
232 | close (sd); | ||
233 | |||
234 | return result; | ||
235 | } | ||
236 | |||
237 | |||
238 | /* opens a connection to a remote host/tcp port */ | ||
239 | int | ||
240 | my_tcp_connect (char *host_name, int port, int *sd) | ||
241 | { | ||
242 | int result; | ||
243 | char proto[4] = "tcp"; | ||
244 | |||
245 | result = my_connect (host_name, port, sd, proto); | ||
246 | |||
247 | return result; | ||
248 | } | ||
249 | |||
250 | |||
251 | /* opens a connection to a remote host/udp port */ | ||
252 | int | ||
253 | my_udp_connect (char *host_name, int port, int *sd) | ||
254 | { | ||
255 | int result; | ||
256 | char proto[4] = "udp"; | ||
257 | |||
258 | result = my_connect (host_name, port, sd, proto); | ||
259 | |||
260 | return result; | ||
261 | } | ||
262 | |||
263 | |||
264 | /* opens a tcp or udp connection to a remote host */ | ||
265 | int | ||
266 | my_connect (char *host_name, int port, int *sd, char *proto) | ||
267 | { | ||
268 | struct sockaddr_in servaddr; | ||
269 | struct hostent *hp; | ||
270 | struct protoent *ptrp; | ||
271 | int result; | ||
272 | |||
273 | bzero ((char *) &servaddr, sizeof (servaddr)); | ||
274 | servaddr.sin_family = AF_INET; | ||
275 | servaddr.sin_port = htons (port); | ||
276 | |||
277 | /* try to bypass using a DNS lookup if this is just an IP address */ | ||
278 | if (!my_inet_aton (host_name, &servaddr.sin_addr)) { | ||
279 | |||
280 | /* else do a DNS lookup */ | ||
281 | hp = gethostbyname ((const char *) host_name); | ||
282 | if (hp == NULL) { | ||
283 | printf ("Invalid host name '%s'\n", host_name); | ||
284 | return STATE_UNKNOWN; | ||
285 | } | ||
286 | |||
287 | memcpy (&servaddr.sin_addr, hp->h_addr, hp->h_length); | ||
288 | } | ||
289 | |||
290 | /* map transport protocol name to protocol number */ | ||
291 | if ((ptrp = getprotobyname (proto)) == NULL) { | ||
292 | printf ("Cannot map \"%s\" to protocol number\n", proto); | ||
293 | return STATE_UNKNOWN; | ||
294 | } | ||
295 | |||
296 | /* create a socket */ | ||
297 | *sd = | ||
298 | socket (PF_INET, (!strcmp (proto, "udp")) ? SOCK_DGRAM : SOCK_STREAM, | ||
299 | ptrp->p_proto); | ||
300 | if (*sd < 0) { | ||
301 | printf ("Socket creation failed\n"); | ||
302 | return STATE_UNKNOWN; | ||
303 | } | ||
304 | |||
305 | /* open a connection */ | ||
306 | result = connect (*sd, (struct sockaddr *) &servaddr, sizeof (servaddr)); | ||
307 | if (result < 0) { | ||
308 | switch (errno) { | ||
309 | case ECONNREFUSED: | ||
310 | printf ("Connection refused by host\n"); | ||
311 | break; | ||
312 | case ETIMEDOUT: | ||
313 | printf ("Timeout while attempting connection\n"); | ||
314 | break; | ||
315 | case ENETUNREACH: | ||
316 | printf ("Network is unreachable\n"); | ||
317 | break; | ||
318 | default: | ||
319 | printf ("Connection refused or timed out\n"); | ||
320 | } | ||
321 | |||
322 | return STATE_CRITICAL; | ||
323 | } | ||
324 | |||
325 | return STATE_OK; | ||
326 | } | ||
327 | |||
328 | |||
329 | |||
330 | /* This code was taken from Fyodor's nmap utility, which was originally | ||
331 | taken from the GLIBC 2.0.6 libraries because Solaris doesn't contain | ||
332 | the inet_aton() funtion. */ | ||
333 | int | ||
334 | my_inet_aton (register const char *cp, struct in_addr *addr) | ||
335 | { | ||
336 | register unsigned int val; /* changed from u_long --david */ | ||
337 | register int base, n; | ||
338 | register char c; | ||
339 | u_int parts[4]; | ||
340 | register u_int *pp = parts; | ||
341 | |||
342 | c = *cp; | ||
343 | |||
344 | for (;;) { | ||
345 | |||
346 | /* | ||
347 | * Collect number up to ``.''. | ||
348 | * Values are specified as for C: | ||
349 | * 0x=hex, 0=octal, isdigit=decimal. | ||
350 | */ | ||
351 | if (!isdigit ((int) c)) | ||
352 | return (0); | ||
353 | val = 0; | ||
354 | base = 10; | ||
355 | |||
356 | if (c == '0') { | ||
357 | c = *++cp; | ||
358 | if (c == 'x' || c == 'X') | ||
359 | base = 16, c = *++cp; | ||
360 | else | ||
361 | base = 8; | ||
362 | } | ||
363 | |||
364 | for (;;) { | ||
365 | if (isascii ((int) c) && isdigit ((int) c)) { | ||
366 | val = (val * base) + (c - '0'); | ||
367 | c = *++cp; | ||
368 | } | ||
369 | else if (base == 16 && isascii ((int) c) && isxdigit ((int) c)) { | ||
370 | val = (val << 4) | (c + 10 - (islower ((int) c) ? 'a' : 'A')); | ||
371 | c = *++cp; | ||
372 | } | ||
373 | else | ||
374 | break; | ||
375 | } | ||
376 | |||
377 | if (c == '.') { | ||
378 | |||
379 | /* | ||
380 | * Internet format: | ||
381 | * a.b.c.d | ||
382 | * a.b.c (with c treated as 16 bits) | ||
383 | * a.b (with b treated as 24 bits) | ||
384 | */ | ||
385 | if (pp >= parts + 3) | ||
386 | return (0); | ||
387 | *pp++ = val; | ||
388 | c = *++cp; | ||
389 | } | ||
390 | else | ||
391 | break; | ||
392 | } | ||
393 | |||
394 | /* Check for trailing characters */ | ||
395 | if (c != '\0' && (!isascii ((int) c) || !isspace ((int) c))) | ||
396 | return (0); | ||
397 | |||
398 | /* Concoct the address according to the number of parts specified */ | ||
399 | n = pp - parts + 1; | ||
400 | switch (n) { | ||
401 | |||
402 | case 0: | ||
403 | return (0); /* initial nondigit */ | ||
404 | |||
405 | case 1: /* a -- 32 bits */ | ||
406 | break; | ||
407 | |||
408 | case 2: /* a.b -- 8.24 bits */ | ||
409 | if (val > 0xffffff) | ||
410 | return (0); | ||
411 | val |= parts[0] << 24; | ||
412 | break; | ||
413 | |||
414 | case 3: /* a.b.c -- 8.8.16 bits */ | ||
415 | if (val > 0xffff) | ||
416 | return (0); | ||
417 | val |= (parts[0] << 24) | (parts[1] << 16); | ||
418 | break; | ||
419 | |||
420 | case 4: /* a.b.c.d -- 8.8.8.8 bits */ | ||
421 | if (val > 0xff) | ||
422 | return (0); | ||
423 | val |= (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8); | ||
424 | break; | ||
425 | } | ||
426 | |||
427 | if (addr) | ||
428 | addr->s_addr = htonl (val); | ||
429 | |||
430 | return (1); | ||
431 | } | ||