/*****************************************************************************
 *
 * Monitoring Plugins network utilities
 *
 * License: GPL
 * Copyright (c) 1999 Ethan Galstad (nagios@nagios.org)
 * Copyright (c) 2003-2024 Monitoring Plugins Development Team
 *
 * Description:
 *
 * This file contains commons functions used in many of the plugins.
 *
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 *
 *
 *****************************************************************************/

#include "common.h"
#include "netutils.h"

unsigned int socket_timeout = DEFAULT_SOCKET_TIMEOUT;
unsigned int socket_timeout_state = STATE_CRITICAL;

int econn_refuse_state = STATE_CRITICAL;
bool was_refused = false;
#if USE_IPV6
int address_family = AF_UNSPEC;
#else
int address_family = AF_INET;
#endif

/* handles socket timeouts */
void socket_timeout_alarm_handler(int sig) {
	if (sig == SIGALRM)
		printf(_("%s - Socket timeout after %d seconds\n"), state_text(socket_timeout_state), socket_timeout);
	else
		printf(_("%s - Abnormal timeout after %d seconds\n"), state_text(socket_timeout_state), socket_timeout);

	exit(socket_timeout_state);
}

/* connects to a host on a specified tcp port, sends a string, and gets a
	 response. loops on select-recv until timeout or eof to get all of a
	 multi-packet answer */
int process_tcp_request2(const char *server_address, int server_port, const char *send_buffer, char *recv_buffer, int recv_size) {

	int result;
	int send_result;
	int recv_result;
	int sd;
	struct timeval tv;
	fd_set readfds;
	int recv_length = 0;

	result = np_net_connect(server_address, server_port, &sd, IPPROTO_TCP);
	if (result != STATE_OK)
		return STATE_CRITICAL;

	send_result = send(sd, send_buffer, strlen(send_buffer), 0);
	if (send_result < 0 || (size_t)send_result != strlen(send_buffer)) {
		printf("%s\n", _("Send failed"));
		result = STATE_WARNING;
	}

	while (1) {
		/* wait up to the number of seconds for socket timeout
		   minus one for data from the host */
		tv.tv_sec = socket_timeout - 1;
		tv.tv_usec = 0;
		FD_ZERO(&readfds);
		FD_SET(sd, &readfds);
		select(sd + 1, &readfds, NULL, NULL, &tv);

		/* make sure some data has arrived */
		if (!FD_ISSET(sd, &readfds)) { /* it hasn't */
			if (!recv_length) {
				strcpy(recv_buffer, "");
				printf("%s\n", _("No data was received from host!"));
				result = STATE_WARNING;
			} else { /* this one failed, but previous ones worked */
				recv_buffer[recv_length] = 0;
			}
			break;
		} else { /* it has */
			recv_result = recv(sd, recv_buffer + recv_length, (size_t)recv_size - recv_length - 1, 0);
			if (recv_result == -1) {
				/* recv failed, bail out */
				strcpy(recv_buffer + recv_length, "");
				result = STATE_WARNING;
				break;
			} else if (recv_result == 0) {
				/* end of file ? */
				recv_buffer[recv_length] = 0;
				break;
			} else { /* we got data! */
				recv_length += recv_result;
				if (recv_length >= recv_size - 1) {
					/* buffer full, we're done */
					recv_buffer[recv_size - 1] = 0;
					break;
				}
			}
		}
		/* end if(!FD_ISSET(sd,&readfds)) */
	}
	/* end while(1) */

	close(sd);
	return result;
}

/* connects to a host on a specified port, sends a string, and gets a
   response */
int process_request(const char *server_address, int server_port, int proto, const char *send_buffer, char *recv_buffer, int recv_size) {
	int result;
	int sd;

	result = STATE_OK;

	result = np_net_connect(server_address, server_port, &sd, proto);
	if (result != STATE_OK)
		return STATE_CRITICAL;

	result = send_request(sd, proto, send_buffer, recv_buffer, recv_size);

	close(sd);

	return result;
}

/* opens a tcp or udp connection to a remote host or local socket */
int np_net_connect(const char *host_name, int port, int *sd, int proto) {
	/* send back STATE_UNKOWN if there's an error
	   send back STATE_OK if we connect
	   send back STATE_CRITICAL if we can't connect.
	   Let upstream figure out what to send to the user. */
	struct addrinfo hints;
	struct addrinfo *r, *res;
	struct sockaddr_un su;
	char port_str[6], host[MAX_HOST_ADDRESS_LENGTH];
	size_t len;
	int socktype, result;
	short is_socket = (host_name[0] == '/');

	socktype = (proto == IPPROTO_UDP) ? SOCK_DGRAM : SOCK_STREAM;

	/* as long as it doesn't start with a '/', it's assumed a host or ip */
	if (!is_socket) {
		memset(&hints, 0, sizeof(hints));
		hints.ai_family = address_family;
		hints.ai_protocol = proto;
		hints.ai_socktype = socktype;

		len = strlen(host_name);
		/* check for an [IPv6] address (and strip the brackets) */
		if (len >= 2 && host_name[0] == '[' && host_name[len - 1] == ']') {
			host_name++;
			len -= 2;
		}
		if (len >= sizeof(host))
			return STATE_UNKNOWN;
		memcpy(host, host_name, len);
		host[len] = '\0';
		snprintf(port_str, sizeof(port_str), "%d", port);
		result = getaddrinfo(host, port_str, &hints, &res);

		if (result != 0) {
			printf("%s\n", gai_strerror(result));
			return STATE_UNKNOWN;
		}

		r = res;
		while (r) {
			/* attempt to create a socket */
			*sd = socket(r->ai_family, socktype, r->ai_protocol);

			if (*sd < 0) {
				printf("%s\n", _("Socket creation failed"));
				freeaddrinfo(r);
				return STATE_UNKNOWN;
			}

			/* attempt to open a connection */
			result = connect(*sd, r->ai_addr, r->ai_addrlen);

			if (result == 0) {
				was_refused = false;
				break;
			}

			if (result < 0) {
				switch (errno) {
				case ECONNREFUSED:
					was_refused = true;
					break;
				}
			}

			close(*sd);
			r = r->ai_next;
		}
		freeaddrinfo(res);
	}
	/* else the hostname is interpreted as a path to a unix socket */
	else {
		if (strlen(host_name) >= UNIX_PATH_MAX) {
			die(STATE_UNKNOWN, _("Supplied path too long unix domain socket"));
		}
		memset(&su, 0, sizeof(su));
		su.sun_family = AF_UNIX;
		strncpy(su.sun_path, host_name, UNIX_PATH_MAX);
		*sd = socket(PF_UNIX, SOCK_STREAM, 0);
		if (*sd < 0) {
			die(STATE_UNKNOWN, _("Socket creation failed"));
		}
		result = connect(*sd, (struct sockaddr *)&su, sizeof(su));
		if (result < 0 && errno == ECONNREFUSED)
			was_refused = true;
	}

	if (result == 0)
		return STATE_OK;
	else if (was_refused) {
		switch (econn_refuse_state) { /* a user-defined expected outcome */
		case STATE_OK:
		case STATE_WARNING:  /* user wants WARN or OK on refusal, or... */
		case STATE_CRITICAL: /* user did not set econn_refuse_state, or wanted critical */
			if (is_socket)
				printf("connect to file socket %s: %s\n", host_name, strerror(errno));
			else
				printf("connect to address %s and port %d: %s\n", host_name, port, strerror(errno));
			return STATE_CRITICAL;
			break;
		default: /* it's a logic error if we do not end up in STATE_(OK|WARNING|CRITICAL) */
			return STATE_UNKNOWN;
			break;
		}
	} else {
		if (is_socket)
			printf("connect to file socket %s: %s\n", host_name, strerror(errno));
		else
			printf("connect to address %s and port %d: %s\n", host_name, port, strerror(errno));
		return STATE_CRITICAL;
	}
}

int send_request(int sd, int proto, const char *send_buffer, char *recv_buffer, int recv_size) {
	int result = STATE_OK;
	int send_result;
	int recv_result;
	struct timeval tv;
	fd_set readfds;

	send_result = send(sd, send_buffer, strlen(send_buffer), 0);
	if (send_result < 0 || (size_t)send_result != strlen(send_buffer)) {
		printf("%s\n", _("Send failed"));
		result = STATE_WARNING;
	}

	/* wait up to the number of seconds for socket timeout minus one
	   for data from the host */
	tv.tv_sec = socket_timeout - 1;
	tv.tv_usec = 0;
	FD_ZERO(&readfds);
	FD_SET(sd, &readfds);
	select(sd + 1, &readfds, NULL, NULL, &tv);

	/* make sure some data has arrived */
	if (!FD_ISSET(sd, &readfds)) {
		strcpy(recv_buffer, "");
		printf("%s\n", _("No data was received from host!"));
		result = STATE_WARNING;
	}

	else {
		recv_result = recv(sd, recv_buffer, (size_t)recv_size - 1, 0);
		if (recv_result == -1) {
			strcpy(recv_buffer, "");
			if (proto != IPPROTO_TCP)
				printf("%s\n", _("Receive failed"));
			result = STATE_WARNING;
		} else
			recv_buffer[recv_result] = 0;

		/* die returned string */
		recv_buffer[recv_size - 1] = 0;
	}
	return result;
}

bool is_host(const char *address) {
	if (is_addr(address) || is_hostname(address))
		return (true);

	return (false);
}

void host_or_die(const char *str) {
	if (!str || (!is_addr(str) && !is_hostname(str)))
		usage_va(_("Invalid hostname/address - %s"), str);
}

bool is_addr(const char *address) {
#ifdef USE_IPV6
	if (address_family == AF_INET && is_inet_addr(address))
		return true;
	else if (address_family == AF_INET6 && is_inet6_addr(address))
		return true;
#else
	if (is_inet_addr(address))
		return (true);
#endif

	return (false);
}

int dns_lookup(const char *in, struct sockaddr_storage *ss, int family) {
	struct addrinfo hints;
	struct addrinfo *res;
	int retval;

	memset(&hints, 0, sizeof(struct addrinfo));
	hints.ai_family = family;

	retval = getaddrinfo(in, NULL, &hints, &res);
	if (retval != 0)
		return false;

	if (ss != NULL)
		memcpy(ss, res->ai_addr, res->ai_addrlen);
	freeaddrinfo(res);
	return true;
}