/* * awesome-client.c - awesome client, communicate with socket * * Copyright © 2007-2008 Julien Danjou * Copyright © 2007 daniel@brinkers.de * * 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 2 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, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * */ #define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include "common/socket.h" #include "common/version.h" #include "common/util.h" /* GNU/Hurd workaround */ #ifndef MSG_NOSIGNAL #define MSG_NOSIGNAL 0 #endif struct sockaddr_un *addr; int csfd; char *display; /** Initialize the client and server socket connections. * If something goes wrong, preserves errno. * \return 0 if everything worked, 1 otherwise. */ static bool sockets_init(void) { if((csfd = socket_getclient()) < 0) return false; if(!(addr = socket_getaddr(display))) return false; if(connect(csfd, addr, sizeof(struct sockaddr_un)) == -1) return false; return true; } /** Close the client and server socket connections. */ static void sockets_close(void) { close(csfd); p_delete(&addr); } /** Reconnect sockets. */ static void sockets_reconnect(void) { warn("connection lost, reconnecting…"); sockets_close(); sockets_init(); } /** Send a message to awesome without handling retries. * \param msg The message. * \param msg_len The message length. * \return The errno of send(). */ static int send_msg_raw(const char *msg, ssize_t msg_len) { #ifndef __FreeBSD__ return send(csfd, msg, msg_len, MSG_NOSIGNAL | MSG_EOR); #else return send(csfd, msg, msg_len, MSG_NOSIGNAL | MSG_EOF); #endif } /** Send a message to awesome. * \param msg The message. * \param msg_len The message length. * \return The errno of send(). */ static int send_msg(const char *msg, ssize_t msg_len) { int try = 10; while (try && send_msg_raw(msg, msg_len) == -1) { switch (errno) { case EPIPE: case ENOTCONN: case ECONNRESET: sockets_reconnect(); try--; break; case ENOENT: warn("can't write to %s", addr->sun_path); return errno; default: warn("error sending packet: %s", strerror(errno)); return errno; } usleep(100000); } if(!try) { warn("giving up."); return EXIT_FAILURE; } return EXIT_SUCCESS; } /** Recieve a message from awesome. */ static void recv_msg(void) { ssize_t r; char buf[1024]; int try = 10; while(try) { r = recv(csfd, buf, sizeof(buf) - 1, MSG_TRUNC | MSG_DONTWAIT); if (r < 0) { if(errno != EAGAIN) return warn("error recieving from UNIX domain socket: %s", strerror(errno)); try--; } else break; usleep(100000); } if(!try) sockets_reconnect(); else if(r > 0) { buf[r] = '\0'; puts(buf); } } /** Print help and exit(2) with given exit_code. * \param exit_code The exit code. * \return Never return. */ static void __attribute__ ((noreturn)) exit_help(int exit_code) { FILE *outfile = (exit_code == EXIT_SUCCESS) ? stdout : stderr; fprintf(outfile, "Usage: awesome-client [--version|--help]\n" "In normal operation, give no parameters and issue commands " "on standard input.\n"); exit(exit_code); } /** Main function of awesome-client. * \param argc Number of args. * \param argv Args array. * \return Value returned by send_msg(). */ int main(int argc, char **argv) { char buf[1024], *msg, *prompt; int ret_value = EXIT_SUCCESS; ssize_t len, msg_len = 1; if(argc == 2) { if(!a_strcmp("-v", argv[1]) || !a_strcmp("--version", argv[1])) eprint_version("awesome-client"); else if(!a_strcmp("-h", argv[1]) || !a_strcmp("--help", argv[1])) exit_help(EXIT_SUCCESS); } else if(argc > 2) exit_help(EXIT_SUCCESS); display = getenv("DISPLAY"); if (!sockets_init()) { warn("can't connect to UNIX domain socket: %s", strerror(errno)); return EXIT_FAILURE; } if(isatty(STDIN_FILENO)) { a_asprintf(&prompt, "awesome@%s%% ", display ? display : "unknown"); while((msg = readline(prompt))) if((msg_len = a_strlen(msg))) { int result; add_history (msg); p_realloc(&msg, msg_len + 2); msg[msg_len] = '\n'; msg[msg_len + 1] = '\0'; result = send_msg(msg, msg_len + 2); if(result == EXIT_SUCCESS) recv_msg(); else if(result == EXIT_FAILURE) break; p_delete(&msg); } } else { msg = p_new(char, 1); while(fgets(buf, sizeof(buf), stdin)) { len = a_strlen(buf); if(len < 2 && msg_len > 1) { ret_value = send_msg(msg, msg_len); p_delete(&msg); if (ret_value != EXIT_SUCCESS) return ret_value; msg = p_new(char, 1); msg_len = 1; } else if (len > 1) { msg_len += len; p_realloc(&msg, msg_len); a_strncat(msg, msg_len, buf, len); } } if(msg_len > 1) { if((ret_value = send_msg(msg, msg_len)) == EXIT_SUCCESS) recv_msg(); } p_delete(&msg); } sockets_close(); return ret_value; } // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80