Add support for action scripts in hostapd_cli

This commit is contained in:
Gregory Detal 2010-04-07 11:14:54 +03:00 committed by Jouni Malinen
parent bb437f282b
commit bae9217474
2 changed files with 146 additions and 21 deletions

View file

@ -3,7 +3,7 @@
hostapd_cli \- hostapd command-line interface hostapd_cli \- hostapd command-line interface
.SH SYNOPSIS .SH SYNOPSIS
.B hostapd_cli .B hostapd_cli
[\-p<path>] [\-i<ifname>] [\-hv] [command..] [\-p<path>] [\-i<ifname>] [\-a<path>] [\-hvB] [command..]
.SH DESCRIPTION .SH DESCRIPTION
This manual page documents briefly the This manual page documents briefly the
.B hostapd_cli .B hostapd_cli
@ -38,6 +38,12 @@ Interface to listen on.
Default: first interface found in socket path. Default: first interface found in socket path.
.TP .TP
.B \-a<path>
Run in daemon mode executing the action file based on events from hostapd.
.TP
.B \-B
Run a daemon in the background.
.TP
.B \-h .B \-h
Show usage. Show usage.
.TP .TP

View file

@ -104,6 +104,8 @@ static int hostapd_cli_quit = 0;
static int hostapd_cli_attached = 0; static int hostapd_cli_attached = 0;
static const char *ctrl_iface_dir = "/var/run/hostapd"; static const char *ctrl_iface_dir = "/var/run/hostapd";
static char *ctrl_ifname = NULL; static char *ctrl_ifname = NULL;
static const char *pid_file = NULL;
static const char *action_file = NULL;
static int ping_interval = 5; static int ping_interval = 5;
@ -112,15 +114,19 @@ static void usage(void)
fprintf(stderr, "%s\n", hostapd_cli_version); fprintf(stderr, "%s\n", hostapd_cli_version);
fprintf(stderr, fprintf(stderr,
"\n" "\n"
"usage: hostapd_cli [-p<path>] [-i<ifname>] [-hv] " "usage: hostapd_cli [-p<path>] [-i<ifname>] [-hvB] "
"[-G<ping interval>] \\\n" "[-a<path>] \\\n"
" [command..]\n" " [-G<ping interval>] [command..]\n"
"\n" "\n"
"Options:\n" "Options:\n"
" -h help (show this usage text)\n" " -h help (show this usage text)\n"
" -v shown version information\n" " -v shown version information\n"
" -p<path> path to find control sockets (default: " " -p<path> path to find control sockets (default: "
"/var/run/hostapd)\n" "/var/run/hostapd)\n"
" -a<file> run in daemon mode executing the action file "
"based on events\n"
" from hostapd\n"
" -B run a daemon in the background\n"
" -i<ifname> Interface to listen on (default: first " " -i<ifname> Interface to listen on (default: first "
"interface found in the\n" "interface found in the\n"
" socket path)\n\n" " socket path)\n\n"
@ -215,6 +221,51 @@ static int hostapd_cli_cmd_mib(struct wpa_ctrl *ctrl, int argc, char *argv[])
} }
static int hostapd_cli_exec(const char *program, const char *arg1,
const char *arg2)
{
char *cmd;
size_t len;
int res;
int ret = 0;
len = os_strlen(program) + os_strlen(arg1) + os_strlen(arg2) + 3;
cmd = os_malloc(len);
if (cmd == NULL)
return -1;
res = os_snprintf(cmd, len, "%s %s %s", program, arg1, arg2);
if (res < 0 || (size_t) res >= len) {
os_free(cmd);
return -1;
}
cmd[len - 1] = '\0';
#ifndef _WIN32_WCE
if (system(cmd) < 0)
ret = -1;
#endif /* _WIN32_WCE */
os_free(cmd);
return ret;
}
static void hostapd_cli_action_process(char *msg, size_t len)
{
const char *pos;
pos = msg;
if (*pos == '<') {
pos = os_strchr(pos, '>');
if (pos)
pos++;
else
pos = msg;
}
hostapd_cli_exec(action_file, ctrl_ifname, pos);
}
static int hostapd_cli_cmd_sta(struct wpa_ctrl *ctrl, int argc, char *argv[]) static int hostapd_cli_cmd_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
{ {
char buf[64]; char buf[64];
@ -558,7 +609,8 @@ static void wpa_request(struct wpa_ctrl *ctrl, int argc, char *argv[])
} }
static void hostapd_cli_recv_pending(struct wpa_ctrl *ctrl, int in_read) static void hostapd_cli_recv_pending(struct wpa_ctrl *ctrl, int in_read,
int action_monitor)
{ {
int first = 1; int first = 1;
if (ctrl_conn == NULL) if (ctrl_conn == NULL)
@ -568,10 +620,14 @@ static void hostapd_cli_recv_pending(struct wpa_ctrl *ctrl, int in_read)
size_t len = sizeof(buf) - 1; size_t len = sizeof(buf) - 1;
if (wpa_ctrl_recv(ctrl, buf, &len) == 0) { if (wpa_ctrl_recv(ctrl, buf, &len) == 0) {
buf[len] = '\0'; buf[len] = '\0';
if (in_read && first) if (action_monitor)
printf("\n"); hostapd_cli_action_process(buf, len);
first = 0; else {
printf("%s\n", buf); if (in_read && first)
printf("\n");
first = 0;
printf("%s\n", buf);
}
} else { } else {
printf("Could not read pending message.\n"); printf("Could not read pending message.\n");
break; break;
@ -589,7 +645,7 @@ static void hostapd_cli_interactive(void)
printf("\nInteractive mode\n\n"); printf("\nInteractive mode\n\n");
do { do {
hostapd_cli_recv_pending(ctrl_conn, 0); hostapd_cli_recv_pending(ctrl_conn, 0, 0);
printf("> "); printf("> ");
alarm(ping_interval); alarm(ping_interval);
res = fgets(cmd, sizeof(cmd), stdin); res = fgets(cmd, sizeof(cmd), stdin);
@ -626,9 +682,19 @@ static void hostapd_cli_interactive(void)
} }
static void hostapd_cli_terminate(int sig) static void hostapd_cli_cleanup(void)
{ {
hostapd_cli_close_connection(); hostapd_cli_close_connection();
if (pid_file)
os_daemonize_terminate(pid_file);
os_program_deinit();
}
static void hostapd_cli_terminate(int sig)
{
hostapd_cli_cleanup();
exit(0); exit(0);
} }
@ -652,25 +718,69 @@ static void hostapd_cli_alarm(int sig)
} }
} }
if (ctrl_conn) if (ctrl_conn)
hostapd_cli_recv_pending(ctrl_conn, 1); hostapd_cli_recv_pending(ctrl_conn, 1, 0);
alarm(ping_interval); alarm(ping_interval);
} }
static void hostapd_cli_action(struct wpa_ctrl *ctrl)
{
fd_set rfds;
int fd, res;
struct timeval tv;
char buf[256];
size_t len;
fd = wpa_ctrl_get_fd(ctrl);
while (!hostapd_cli_quit) {
FD_ZERO(&rfds);
FD_SET(fd, &rfds);
tv.tv_sec = ping_interval;
tv.tv_usec = 0;
res = select(fd + 1, &rfds, NULL, NULL, &tv);
if (res < 0 && errno != EINTR) {
perror("select");
break;
}
if (FD_ISSET(fd, &rfds))
hostapd_cli_recv_pending(ctrl, 0, 1);
else {
len = sizeof(buf) - 1;
if (wpa_ctrl_request(ctrl, "PING", 4, buf, &len,
hostapd_cli_action_process) < 0 ||
len < 4 || os_memcmp(buf, "PONG", 4) != 0) {
printf("hostapd did not reply to PING "
"command - exiting\n");
break;
}
}
}
}
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
int interactive; int interactive;
int warning_displayed = 0; int warning_displayed = 0;
int c; int c;
int daemonize = 0;
if (os_program_init()) if (os_program_init())
return -1; return -1;
for (;;) { for (;;) {
c = getopt(argc, argv, "hG:i:p:v"); c = getopt(argc, argv, "a:BhG:i:p:v");
if (c < 0) if (c < 0)
break; break;
switch (c) { switch (c) {
case 'a':
action_file = optarg;
break;
case 'B':
daemonize = 1;
break;
case 'G': case 'G':
ping_interval = atoi(optarg); ping_interval = atoi(optarg);
break; break;
@ -693,7 +803,7 @@ int main(int argc, char *argv[])
} }
} }
interactive = argc == optind; interactive = (argc == optind) && (action_file == NULL);
if (interactive) { if (interactive) {
printf("%s\n\n%s\n\n", hostapd_cli_version, printf("%s\n\n%s\n\n", hostapd_cli_version,
@ -742,18 +852,27 @@ int main(int argc, char *argv[])
signal(SIGTERM, hostapd_cli_terminate); signal(SIGTERM, hostapd_cli_terminate);
signal(SIGALRM, hostapd_cli_alarm); signal(SIGALRM, hostapd_cli_alarm);
if (interactive) { if (interactive || action_file) {
if (wpa_ctrl_attach(ctrl_conn) == 0) { if (wpa_ctrl_attach(ctrl_conn) == 0) {
hostapd_cli_attached = 1; hostapd_cli_attached = 1;
} else { } else {
printf("Warning: Failed to attach to hostapd.\n"); printf("Warning: Failed to attach to hostapd.\n");
if (action_file)
return -1;
} }
}
if (daemonize && os_daemonize(pid_file))
return -1;
if (interactive)
hostapd_cli_interactive(); hostapd_cli_interactive();
} else else if (action_file)
hostapd_cli_action(ctrl_conn);
else
wpa_request(ctrl_conn, argc - optind, &argv[optind]); wpa_request(ctrl_conn, argc - optind, &argv[optind]);
free(ctrl_ifname); os_free(ctrl_ifname);
hostapd_cli_close_connection(); hostapd_cli_cleanup();
os_program_deinit();
return 0; return 0;
} }