summaryrefslogblamecommitdiffstats
path: root/src/client/client.c
blob: 80c6403f12c1399db6dfb5512b22700d01435c02 (plain) (tree)



















                                                               






                       

                      
                   
 
                     

                       
                                   
 


                                                                                                                       
                                                                                            









                                                                                                

 
                          
 

                                                

 









                                                                                                               
 



























































                                                    
 






                                                            




                                                               




                                                     

            
                                                                              

                                   
                                 

 

                                









                                                  

                                    


























                                                                       

                              
                                                        



















                                                           
                                                        















                                                                               
                                                                            















                                                                                                      
                                                                              















                                                                                                       
                                                               















                                                                                                        

                                  
 
/*
 * This file is part of the Distributed Network Block Device 3
 *
 * Copyright(c) 2011-2012 Johann Latocha <johann@latocha.de>
 *
 * This file may be licensed under the terms of of the
 * GNU General Public License Version 2 (the ``GPL'').
 *
 * Software distributed under the License is distributed
 * on an ``AS IS'' basis, WITHOUT WARRANTY OF ANY KIND, either
 * express or implied. See the GPL for the specific language
 * governing rights and limitations.
 *
 * You should have received a copy of the GPL along with this
 * program. If not, go to http://www.gnu.org/licenses/gpl.html
 * or write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 *
 */

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>
#include <getopt.h>
#include <sys/ioctl.h>
#include <netinet/in.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <string.h>

#include "../types.h"
#include "../version.h"

void dnbd3_print_help(char *argv_0)
{
	printf("\nUsage: %s\n"
	       "\t-h <host> -i <image name> [-r <rid>] -d <device> [-a <KB>] || -f <file> || -c <device>\n\n", argv_0);
	printf("Start the DNBD3 client.\n");
	//printf("-f or --file \t\t Configuration file (default /etc/dnbd3-client.conf)\n");
	printf("-h or --host \t\t Host running dnbd3-server.\n");
	printf("-i or --image \t\t Image name of exported image.\n");
	printf("-r or --rid \t\t Release-ID of exported image (default 0, latest).\n");
	printf("-d or --device \t\t DNBD3 device name.\n");
	printf("-a or --ahead \t\t Read ahead in KByte (default %i).\n", DEFAULT_READ_AHEAD_KB);
	printf("-c or --close \t\t Disconnect and close device.\n");
	printf("-s or --switch \t\t Switch dnbd3-server on device (DEBUG).\n");
	printf("-H or --help \t\t Show this help text and quit.\n");
	printf("-V or --version \t Show version and quit.\n\n");
	exit(EXIT_SUCCESS);
}

void dnbd3_print_version()
{
	printf("Version: %s\n", VERSION_STRING);
	exit(EXIT_SUCCESS);
}

/**
 * Parse IPv4 or IPv6 address in string representation to a suitable format usable by the BSD socket library
 * @string eg. "1.2.3.4" or "2a01::10:5", optially with port appended, eg "1.2.3.4:6666" or "[2a01::10:5]:6666"
 * @af will contain either AF_INET or AF_INET6
 * @addr will contain the address in network representation
 * @port will contain the port in network representation, defaulting to #define PORT if none was given
 * returns 1 on success, 0 in failure. contents of af, addr and port are undefined in the latter case
 * !! Contents of @string might be modified by this function !!
 */
static char parse_address(char *string, dnbd3_host_t *host)
{
	struct in_addr v4;
	struct in6_addr v6;

	// Try IPv4 without port
	if (1 == inet_pton(AF_INET, string, &v4))
	{
		host->type = AF_INET;
		memcpy(host->addr, &v4, 4);
		host->port = htons(PORT);
		return 1;
	}
	// Try IPv6 without port
	if (1 == inet_pton(AF_INET6, string, &v6))
	{
		host->type = AF_INET6;
		memcpy(host->addr, &v6, 16);
		host->port = htons(PORT);
		return 1;
	}

	// Scan for port
	char *portpos = NULL, *ptr = string;
	while (*ptr)
	{
		if (*ptr == ':')
			portpos = ptr;
		++ptr;
	}
	if (portpos == NULL)
		return 0; // No port in string
	// Consider IP being surrounded by [ ]
	if (*string == '[' && *(portpos - 1) == ']')
	{
		++string;
		*(portpos - 1) = '\0';
	}
	*portpos++ = '\0';
	int p = atoi(portpos);
	if (p < 1 || p > 65535)
		return 0; // Invalid port
	host->port = htons((uint16_t)p);

	// Try IPv4 with port
	if (1 == inet_pton(AF_INET, string, &v4))
	{
		host->type = AF_INET;
		memcpy(host->addr, &v4, 4);
		return 1;
	}
	// Try IPv6 with port
	if (1 == inet_pton(AF_INET6, string, &v6))
	{
		host->type = AF_INET6;
		memcpy(host->addr, &v6, 16);
		return 1;
	}

	// FAIL
	return 0;
}

static void dnbd3_get_ip(char *hostname, dnbd3_host_t *host)
{
	if (parse_address(hostname, host))
		return;
	// TODO: Parse port too for host names
	struct hostent *hent;
	if ((hent = gethostbyname(hostname)) == NULL)
	{
		printf("FATAL: Unknown host '%s'\n", hostname);
		exit(EXIT_FAILURE);
	}

	host->type = (uint8_t)hent->h_addrtype;
	if (hent->h_addrtype == AF_INET)
		memcpy(host->addr, hent->h_addr, 4);
	else if (hent->h_addrtype == AF_INET6)
		memcpy(host->addr, hent->h_addr, 16);
	else
	{
		printf("FATAL: Unknown address type: %d\n", hent->h_addrtype);
		exit(EXIT_FAILURE);
	}
	host->port = htons(PORT);
}

int main(int argc, char *argv[])
{
	int fd;
	char *dev = NULL;

	int close_dev = 0;
	int switch_host = 0;

	dnbd3_ioctl_t msg;
	memset(&msg, 0, sizeof(dnbd3_ioctl_t));
	msg.len = (uint16_t)sizeof(dnbd3_ioctl_t);
	msg.read_ahead_kb = DEFAULT_READ_AHEAD_KB;
	msg.host.port = htons(PORT);
	msg.host.type = 0;
	msg.imgname = NULL;
	msg.is_server = FALSE;

	int opt = 0;
	int longIndex = 0;
	static const char *optString = "f:h:i:r:d:a:c:s:HV?";
	static const struct option longOpts[] =
	{
		{ "file", required_argument, NULL, 'f' },
		{ "host", required_argument, NULL, 'h' },
		{ "image", required_argument, NULL, 'i' },
		{ "rid", required_argument, NULL, 'r' },
		{ "device", required_argument, NULL, 'd' },
		{ "ahead", required_argument, NULL, 'a' },
		{ "close", required_argument, NULL, 'c' },
		{ "switch", required_argument, NULL, 's' },
		{ "help", no_argument, NULL, 'H' },
		{ "version", no_argument, NULL, 'V' },
	};

	opt = getopt_long(argc, argv, optString, longOpts, &longIndex);

	while (opt != -1)
	{
		switch (opt)
		{
		case 'f':
			break;
		case 'h':
			dnbd3_get_ip(optarg, &msg.host);
			break;
		case 'i':
			msg.imgname = strdup(optarg);
			printf("Image: %s\n", msg.imgname);
			break;
		case 'r':
			msg.rid = atoi(optarg);
			break;
		case 'd':
			dev = strdup(optarg);
			printf("Device is %s\n", dev);
			break;
		case 'a':
			msg.read_ahead_kb = atoi(optarg);
			break;
		case 'c':
			dev = strdup(optarg);
			close_dev = 1;
			break;
		case 's':
			dnbd3_get_ip(optarg, &msg.host);
			switch_host = 1;
			break;
		case 'H':
			dnbd3_print_help(argv[0]);
			break;
		case 'V':
			dnbd3_print_version();
			break;
		case '?':
			dnbd3_print_help(argv[0]);
			break;
		}
		opt = getopt_long(argc, argv, optString, longOpts, &longIndex);
	}

	// close device
	if (close_dev && msg.host.type == 0 && dev && (msg.imgname == NULL))
	{
		fd = open(dev, O_WRONLY);
		printf("INFO: Closing device %s\n", dev);

		const int ret = ioctl(fd, IOCTL_CLOSE, &msg);
		if (ret < 0)
		{
			printf("ERROR: ioctl not successful (close, %s (%d))\n", strerror(-ret), ret);
			exit(EXIT_FAILURE);
		}

		close(fd);
		exit(EXIT_SUCCESS);
	}

	// switch host
	if (switch_host && msg.host.type != 0 && dev && (msg.imgname == NULL))
	{
		fd = open(dev, O_WRONLY);
		printf("INFO: Switching device %s to %s\n", dev, "<fixme>");

		const int ret = ioctl(fd, IOCTL_SWITCH, &msg);
		if (ret < 0)
		{
			printf("ERROR: ioctl not successful (switch, %s (%d))\n", strerror(-ret), ret);
			exit(EXIT_FAILURE);
		}

		close(fd);
		exit(EXIT_SUCCESS);
	}

	// connect
	if (msg.host.type != 0 && dev && (msg.imgname != NULL))
	{
		msg.imgnamelen = (uint16_t)strlen(msg.imgname);
		fd = open(dev, O_WRONLY);
		printf("INFO: Connecting %s to %s (%s rid:%i)\n", dev, "<fixme>", msg.imgname, msg.rid);

		const int ret = ioctl(fd, IOCTL_OPEN, &msg);
		if (ret < 0)
		{
			printf("ERROR: ioctl not successful (connect, %s (%d))\n", strerror(-ret), ret);
			exit(EXIT_FAILURE);
		}

		close(fd);
		exit(EXIT_SUCCESS);
	}

	dnbd3_print_help(argv[0]);
	exit(EXIT_FAILURE);
}