
Add the --help option Add quoting for executing command with arguments that contain whitespace
532 lines
11 KiB
C
532 lines
11 KiB
C
/*
|
|
* This file is distributed as part of MaxScale. It 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,
|
|
* version 2.
|
|
*
|
|
* 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, write to the Free Software Foundation, Inc., 51
|
|
* Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Copyright SkySQL Ab 2014
|
|
*/
|
|
|
|
/**
|
|
* @file maxadmin.c - The MaxScale administration client
|
|
*
|
|
* @verbatim
|
|
* Revision History
|
|
*
|
|
* Date Who Description
|
|
* 13/06/14 Mark Riddoch Initial implementation
|
|
* 15/06/14 Mark Riddoch Addition of source command
|
|
*
|
|
* @endverbatim
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <netdb.h>
|
|
#include <ctype.h>
|
|
#include <stdlib.h>
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
#include <dirent.h>
|
|
#include <locale.h>
|
|
#include <errno.h>
|
|
|
|
#ifdef HISTORY
|
|
#include <histedit.h>
|
|
#endif
|
|
|
|
static int connectMaxScale(char *hostname, char *port);
|
|
static int setipaddress(struct in_addr *a, char *p);
|
|
static int authMaxScale(int so, char *user, char *password);
|
|
static int sendCommand(int so, char *cmd);
|
|
static void DoSource(int so, char *cmd);
|
|
static void DoUsage();
|
|
|
|
#ifdef HISTORY
|
|
static char *
|
|
prompt(EditLine *el __attribute__((__unused__)))
|
|
{
|
|
static char prompt[] = "MaxScale> ";
|
|
|
|
return prompt;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* The main for the maxadmin client
|
|
*
|
|
* @param argc Number of arguments
|
|
* @param argv The command line arguments
|
|
*/
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
int i, num, rv, fatal = 0;
|
|
#ifdef HISTORY
|
|
char *buf;
|
|
EditLine *el = NULL;
|
|
Tokenizer *tok;
|
|
History *hist;
|
|
HistEvent ev;
|
|
const LineInfo *li;
|
|
#else
|
|
char buf[1024];
|
|
#endif
|
|
char *hostname = "localhost";
|
|
char *port = "6603";
|
|
char *user = "admin";
|
|
char *passwd = NULL;
|
|
int so, cmdlen;
|
|
char *cmd;
|
|
int argno = 0;
|
|
|
|
cmd = malloc(1);
|
|
*cmd = 0;
|
|
cmdlen = 1;
|
|
|
|
for (i = 1; i < argc; i++)
|
|
{
|
|
if (argv[i][0] == '-')
|
|
{
|
|
switch (argv[i][1])
|
|
{
|
|
case 'u': /* User */
|
|
if (argv[i][2])
|
|
user = &(argv[i][2]);
|
|
else if (i + 1 < argc)
|
|
user = argv[++i];
|
|
else
|
|
{
|
|
fprintf(stderr, "Missing username"
|
|
"in -u option.\n");
|
|
fatal = 1;
|
|
}
|
|
break;
|
|
case 'p': /* Password */
|
|
if (argv[i][2])
|
|
passwd = &(argv[i][2]);
|
|
else if (i + 1 < argc)
|
|
passwd = argv[++i];
|
|
else
|
|
{
|
|
fprintf(stderr, "Missing password "
|
|
"in -p option.\n");
|
|
fatal = 1;
|
|
}
|
|
break;
|
|
case 'h': /* hostname */
|
|
if (argv[i][2])
|
|
hostname = &(argv[i][2]);
|
|
else if (i + 1 < argc)
|
|
hostname = argv[++i];
|
|
else
|
|
{
|
|
fprintf(stderr, "Missing hostname value "
|
|
"in -h option.\n");
|
|
fatal = 1;
|
|
}
|
|
break;
|
|
case 'P': /* Port */
|
|
if (argv[i][2])
|
|
port = &(argv[i][2]);
|
|
else if (i + 1 < argc)
|
|
port = argv[++i];
|
|
else
|
|
{
|
|
fprintf(stderr, "Missing Port value "
|
|
"in -P option.\n");
|
|
fatal = 1;
|
|
}
|
|
break;
|
|
case '-':
|
|
{
|
|
char *word;
|
|
|
|
word = &argv[i][2];
|
|
if (strcmp(word, "help") == 0)
|
|
{
|
|
DoUsage();
|
|
exit(0);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Arguments after the second argument are quoted
|
|
* to allow for quoted names on the command line
|
|
* to be passed on in quotes.
|
|
*/
|
|
if (argno++ > 1)
|
|
{
|
|
cmdlen += strlen(argv[i]) + 3;
|
|
cmd = realloc(cmd, cmdlen);
|
|
strcat(cmd, "\"");
|
|
strcat(cmd, argv[i]);
|
|
strcat(cmd, "\" ");
|
|
}
|
|
else
|
|
{
|
|
cmdlen += strlen(argv[i]) + 1;
|
|
cmd = realloc(cmd, cmdlen);
|
|
strcat(cmd, argv[i]);
|
|
strcat(cmd, " ");
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fatal)
|
|
exit(1);
|
|
|
|
if (passwd == NULL)
|
|
{
|
|
struct termios tty_attr;
|
|
tcflag_t c_lflag;
|
|
|
|
if (tcgetattr(STDIN_FILENO, &tty_attr) < 0)
|
|
return -1;
|
|
|
|
c_lflag = tty_attr.c_lflag;
|
|
tty_attr.c_lflag &= ~ICANON;
|
|
tty_attr.c_lflag &= ~ECHO;
|
|
|
|
if (tcsetattr(STDIN_FILENO, 0, &tty_attr) < 0)
|
|
return -1;
|
|
|
|
printf("Password: ");
|
|
passwd = malloc(80);
|
|
fgets(passwd, 80, stdin);
|
|
|
|
tty_attr.c_lflag = c_lflag;
|
|
if (tcsetattr(STDIN_FILENO, 0, &tty_attr) < 0)
|
|
return -1;
|
|
i = strlen(passwd);
|
|
if (i > 1)
|
|
passwd[i - 1] = '\0';
|
|
printf("\n");
|
|
}
|
|
|
|
if ((so = connectMaxScale(hostname, port)) == -1)
|
|
exit(1);
|
|
if (!authMaxScale(so, user, passwd))
|
|
{
|
|
fprintf(stderr, "Failed to connect to MaxScale. "
|
|
"Incorrect username or password.\n");
|
|
exit(1);
|
|
}
|
|
|
|
if (cmdlen > 1)
|
|
{
|
|
cmd[cmdlen - 2] = '\0'; /* Remove trailing space */
|
|
if (access(cmd, R_OK) == 0)
|
|
DoSource(so, cmd);
|
|
else
|
|
sendCommand(so, cmd);
|
|
exit(0);
|
|
}
|
|
|
|
(void) setlocale(LC_CTYPE, "");
|
|
#ifdef HISTORY
|
|
hist = history_init(); /* Init the builtin history */
|
|
/* Remember 100 events */
|
|
history(hist, &ev, H_SETSIZE, 100);
|
|
|
|
tok = tok_init(NULL); /* Initialize the tokenizer */
|
|
|
|
/* Initialize editline */
|
|
el = el_init(*argv, stdin, stdout, stderr);
|
|
|
|
el_set(el, EL_EDITOR, "vi"); /* Default editor is vi */
|
|
el_set(el, EL_SIGNAL, 1); /* Handle signals gracefully */
|
|
el_set(el, EL_PROMPT, prompt);/* Set the prompt function */
|
|
|
|
/* Tell editline to use this history interface */
|
|
el_set(el, EL_HIST, history, hist);
|
|
|
|
/*
|
|
* Bind j, k in vi command mode to previous and next line, instead
|
|
* of previous and next history.
|
|
*/
|
|
el_set(el, EL_BIND, "-a", "k", "ed-prev-line", NULL);
|
|
el_set(el, EL_BIND, "-a", "j", "ed-next-line", NULL);
|
|
|
|
/*
|
|
* Source the user's defaults file.
|
|
*/
|
|
el_source(el, NULL);
|
|
|
|
while ((buf = el_gets(el, &num)) != NULL && num != 0)
|
|
{
|
|
#else
|
|
while (printf("MaxScale> ") && fgets(buf, 1024, stdin) != NULL)
|
|
{
|
|
num = strlen(buf);
|
|
#endif
|
|
/* Strip trailing \n\r */
|
|
for (i = num - 1; buf[i] == '\r' || buf[i] == '\n'; i--)
|
|
buf[i] = 0;
|
|
|
|
#ifdef HISTORY
|
|
li = el_line(el);
|
|
history(hist, &ev, H_ENTER, buf);
|
|
#endif
|
|
|
|
if (!strcasecmp(buf, "quit"))
|
|
{
|
|
break;
|
|
}
|
|
else if (!strcasecmp(buf, "history"))
|
|
{
|
|
#ifdef HISTORY
|
|
for (rv = history(hist, &ev, H_LAST); rv != -1;
|
|
rv = history(hist, &ev, H_PREV))
|
|
fprintf(stdout, "%4d %s\n",
|
|
ev.num, ev.str);
|
|
#else
|
|
fprintf(stderr, "History not supported in this version.\n");
|
|
#endif
|
|
}
|
|
else if (!strncasecmp(buf, "source", 6))
|
|
{
|
|
char *ptr;
|
|
|
|
/* Find the filename */
|
|
ptr = &buf[strlen("source")];
|
|
while (*ptr && isspace(*ptr))
|
|
ptr++;
|
|
|
|
DoSource(so, ptr);
|
|
}
|
|
else if (*buf)
|
|
{
|
|
sendCommand(so, buf);
|
|
}
|
|
}
|
|
|
|
#ifdef HISTORY
|
|
el_end(el);
|
|
tok_end(tok);
|
|
history_end(hist);
|
|
#endif
|
|
close(so);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Connect to the MaxScale server
|
|
*
|
|
* @param hostname The hostname to connect to
|
|
* @param port The port to use for the connection
|
|
* @return The connected socket or -1 on error
|
|
*/
|
|
static int
|
|
connectMaxScale(char *hostname, char *port)
|
|
{
|
|
struct sockaddr_in addr;
|
|
int so;
|
|
|
|
if ((so = socket(AF_INET, SOCK_STREAM, 0)) < 0)
|
|
{
|
|
fprintf(stderr, "Unable to create socket: %s\n",
|
|
strerror(errno));
|
|
return -1;
|
|
}
|
|
memset(&addr, 0, sizeof addr);
|
|
addr.sin_family = AF_INET;
|
|
setipaddress(&addr.sin_addr, hostname);
|
|
addr.sin_port = htons(atoi(port));
|
|
if (connect(so, (struct sockaddr *)&addr, sizeof(addr)) < 0)
|
|
{
|
|
fprintf(stderr, "Unable to connect to MaxScale at %s, %s: %s\n",
|
|
hostname, port, strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
return so;
|
|
}
|
|
|
|
|
|
/**
|
|
* Set IP address in socket structure in_addr
|
|
*
|
|
* @param a Pointer to a struct in_addr into which the address is written
|
|
* @param p The hostname to lookup
|
|
* @return 1 on success, 0 on failure
|
|
*/
|
|
static int
|
|
setipaddress(struct in_addr *a, char *p)
|
|
{
|
|
#ifdef __USE_POSIX
|
|
struct addrinfo *ai = NULL, hint;
|
|
int rc;
|
|
struct sockaddr_in * res_addr;
|
|
memset(&hint, 0, sizeof (hint));
|
|
|
|
hint.ai_socktype = SOCK_STREAM;
|
|
hint.ai_flags = AI_CANONNAME;
|
|
hint.ai_family = AF_INET;
|
|
|
|
if ((rc = getaddrinfo(p, NULL, &hint, &ai)) != 0) {
|
|
return 0;
|
|
}
|
|
|
|
/* take the first one */
|
|
if (ai != NULL) {
|
|
res_addr = (struct sockaddr_in *)(ai->ai_addr);
|
|
memcpy(a, &res_addr->sin_addr, sizeof(struct in_addr));
|
|
|
|
freeaddrinfo(ai);
|
|
|
|
return 1;
|
|
}
|
|
#else
|
|
struct hostent *h;
|
|
|
|
spinlock_acquire(&tmplock);
|
|
h = gethostbyname(p);
|
|
spinlock_release(&tmplock);
|
|
|
|
if (h == NULL) {
|
|
if ((a->s_addr = inet_addr(p)) == -1) {
|
|
return 0;
|
|
}
|
|
} else {
|
|
/* take the first one */
|
|
memcpy(a, h->h_addr, h->h_length);
|
|
|
|
return 1;
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform authentication using the maxscaled protocol conventions
|
|
*
|
|
* @param so The socket connected to MaxScale
|
|
* @param user The username to authenticate
|
|
* @param password The password to authenticate with
|
|
* @return Non-zero of succesful authentication
|
|
*/
|
|
static int
|
|
authMaxScale(int so, char *user, char *password)
|
|
{
|
|
char buf[20];
|
|
|
|
read(so, buf, 4);
|
|
write(so, user, strlen(user));
|
|
read(so, buf, 8);
|
|
write(so, password, strlen(password));
|
|
read(so, buf, 6);
|
|
|
|
return strncmp(buf, "FAILED", 6);
|
|
}
|
|
|
|
/**
|
|
* Send a comamnd using the MaxScaled protocol, display the return data
|
|
* on standard output
|
|
*
|
|
* @param so The socket connect to MaxScale
|
|
* @param cmd The command to send
|
|
* @return 0 if the connection was closed
|
|
*/
|
|
static int
|
|
sendCommand(int so, char *cmd)
|
|
{
|
|
char buf[80];
|
|
int i;
|
|
|
|
write(so, cmd, strlen(cmd));
|
|
while (1)
|
|
{
|
|
if ((i = read(so, buf, 80)) == -1)
|
|
return 0;
|
|
if (i > 1 && buf[i-1] == 'K' && buf[i-2] == 'O')
|
|
{
|
|
write(1, buf, i - 2);
|
|
return 1;
|
|
}
|
|
write(1, buf, i);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* Read a file of commands and send them to MaxScale
|
|
*
|
|
* @param so The socket connected to MaxScale
|
|
* @param file The filename
|
|
*/
|
|
static void
|
|
DoSource(int so, char *file)
|
|
{
|
|
char *ptr, *pe;
|
|
char line[132];
|
|
FILE *fp;
|
|
|
|
if ((fp = fopen(file, "r")) == NULL)
|
|
{
|
|
fprintf(stderr, "Unable to open command file '%s'.\n",
|
|
file);
|
|
return;
|
|
}
|
|
|
|
while ((ptr = fgets(line, 132, fp)) != NULL)
|
|
{
|
|
/* Strip tailing newlines */
|
|
pe = &ptr[strlen(ptr)-1];
|
|
while (pe >= ptr && (*pe == '\r' || *pe == '\n'))
|
|
{
|
|
*pe = '\0';
|
|
pe--;
|
|
}
|
|
|
|
if (*ptr != '#') /* Comment */
|
|
{
|
|
if (! sendCommand(so, ptr))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
fclose(fp);
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Display the --help text.
|
|
*/
|
|
static void
|
|
DoUsage()
|
|
{
|
|
printf("maxadmin: The MaxScale administrative and monitor client.\n\n");
|
|
printf("Usage: maxadmin [-u user] [-p password] [-h hostname] [-P port] [<command file> | <command>]\n\n");
|
|
printf(" -u user The user name to use for the connection, default\n");
|
|
printf(" is admin.\n");
|
|
printf(" -p password The user password, if not given the password will\n");
|
|
printf(" be prompted for interactively\n");
|
|
printf(" -h hostname The maxscale host to connecto to. The default is\n");
|
|
printf(" localhost\n");
|
|
printf(" -P port The port to use for the connection, the default\n");
|
|
printf(" port is 6603.\n");
|
|
printf(" --help Print this help text.\n");
|
|
printf("Any remaining arguments are treated as MaxScale commands or a file\n");
|
|
printf("containing commands to execute.\n");
|
|
}
|