/* * client_commands.c * This file is part of LCDd, the lcdproc server. * * This file is released under the GNU General Public License. Refer to the * COPYING file distributed with this package. * * Copyright (c) 1999, William Ferrell, Scott Scriven * 2002, Joris Robijn * * * This contains definitions for all the functions which clients can run. * The functions here are to be called only from parse.c's interpreter. * * The client's available function set is defined here, as is the syntax * for each command. * * This particular file defines actions concerning clients. * */ #include #include #include #include #include #include #include "shared/report.h" #include "shared/sockets.h" #include "drivers.h" #include "render.h" #include "client.h" #include "input.h" /*************************************************************** * Debugging only.. prints out a list of arguments it receives */ int test_func_func (Client * c, int argc, char **argv) { int i; char str[256]; for (i = 0; i < argc; i++) { snprintf (str, sizeof(str), "%s: %i -> %s\n", __FUNCTION__, i, argv[i]); report (RPT_INFO, "%s", str); sock_send_string (c->sock, str); } return 0; } /*************************************************************** * The client must say "hello" before doing anything else. * * It returns a string of info about the server to the client * * Usage: hello */ int hello_func (Client * c, int argc, char **argv) { char str[256]; /* TODO: Give *real* info about the server/lcd...*/ if (argc > 1) { sock_send_error(c->sock, "extra parameters ignored\n"); } debug(RPT_INFO, "Hello!"); memset(str, '\0', sizeof(str)); snprintf(str, sizeof(str), "connect LCDproc %s protocol %s lcd wid %i hgt %i cellwid %i cellhgt %i\n", VERSION, PROTOCOL_VERSION, display_props->width, display_props->height, display_props->cellwidth, display_props->cellheight); sock_send_string (c->sock, str); c->ack = 1; return 0; } /***************************************************************************** * The client should say "bye" before disconnecting * * The function does not respond to the client: it simply cuts connection * * Usage: bye */ int bye_func (Client * c, int argc, char **argv) { if (c != NULL) { debug(RPT_INFO, "Bye, %s!", (c->name != NULL) ? c->name : "unknown client"); sock_send_error(c->sock, "\"bye\" is currently ignored\n"); } return 0; } /*************************************************** * sets info about the client, such as its name * * Usage: client_set -name */ int client_set_func (Client * c, int argc, char **argv) { int i; char str[256]; memset(str, '\0', sizeof(str)); if (!c->ack) return 1; if (argc != 3) { sock_send_error(c->sock, "Usage: client_set -name \n"); return 0; } i = 1; do { char *p; /* This bit of code means that "-name" is the same as "name"...*/ p = argv[i]; if (*p == '-') p++; /* Handle the "name" parameter*/ if (strcmp (p, "name") == 0) { i++; if (argv[i] == '\0') { sock_printf_error(c->sock, "internal error: no parameter #%d\n", i); continue; } if (strlen(argv[i]) > sizeof(str) -1) { sock_send_error(c->sock, "name too long\n"); } else { strncpy(str, argv[i], sizeof(str) - 1); debug(RPT_DEBUG, "client_set: name=\"%s\"", argv[i]); /* set the name...*/ if (c->name) free (c->name); if ((c->name = strdup (str)) == NULL) { sock_send_error(c->sock, "error allocating memory!\n"); } else { sock_send_string(c->sock, "success\n"); i++; /* bypass argument (name string)*/ } } } else { sock_printf_error(c->sock, "invalid parameter (%s)\n", p); } } while (++i < argc); return 0; } /****************************************************************** * Tells the server the client would like to accept keypresses * of a particular type * * Usage: client_add_key [-exclusively|-shared] {}+ */ #define BUFLEN 80 int client_add_key_func (Client * c, int argc, char **argv) { int exclusively = 0; int argnr; if (!c->ack) return 1; if (argc < 2) { sock_send_error(c->sock, "Usage: client_add_key [-exclusively|-shared] {}+\n"); return 0; } argnr = 1; if( argv[argnr][0] == '-' ) { if( strcmp( argv[argnr], "-shared") == 0 ) { exclusively = 0; } else if( strcmp( argv[argnr], "-exclusively") == 0 ) { exclusively = 1; } else { sock_printf_error(c->sock, "Invalid option: %s\n", argv[argnr]); } argnr ++; } for ( ; argnr < argc; argnr++ ) { if( input_reserve_key( argv[argnr], exclusively, c ) < 0 ) { sock_printf_error(c->sock, "Could not reserve key \"%s\"\n", argv[argnr]); } } sock_send_string(c->sock, "success\n"); return 0; } /********************************************************************* * Tells the server the client would NOT like to accept keypresses * of a particular type * * Usage: client_del_key {}+ */ int client_del_key_func (Client * c, int argc, char **argv) { int argnr; if (!c->ack) return 1; if (argc < 2) { sock_send_error(c->sock, "Usage: client_del_key {}+\n"); return 0; } for( argnr=1; argnr < argc; argnr++) { input_release_key( argv[argnr], c ); } sock_send_string(c->sock, "success\n"); return 0; } /*************************************************************************** * Toggles the backlight, if enabled. * * Usage: backlight {on|off|toggle|blink|flash} */ int backlight_func (Client * c, int argc, char **argv) { if (!c->ack) return 1; if (argc != 2) { sock_send_error(c->sock, "Usage: backlight {on|off|toggle|blink|flash}\n"); return 0; } debug (RPT_DEBUG, "backlight(%s)", argv[1]); //backlight = (backlight && 1); /* only preserves ON/OFF bit*/ if (strcmp ("on", argv[1]) == 0) { c->backlight = BACKLIGHT_ON; } else if (strcmp ("off", argv[1]) == 0) { c->backlight = BACKLIGHT_OFF; } else if (strcmp ("toggle", argv[1]) == 0) { if (c->backlight == BACKLIGHT_ON) c->backlight = BACKLIGHT_OFF; else if (c->backlight == BACKLIGHT_OFF) c->backlight = BACKLIGHT_ON; } else if (strcmp ("blink", argv[1]) == 0) { c->backlight |= BACKLIGHT_BLINK; } else if (strcmp ("flash", argv[1]) == 0) { c->backlight |= BACKLIGHT_FLASH; } sock_send_string(c->sock, "success\n"); return 0; } /**************************************************************************** * info_func * * Usage: info */ int info_func (Client * c, int argc, char **argv) { char str[1024]; if (argc > 1) { sock_send_error(c->sock, "Extra arguments ignored...\n"); } memset(str, '\0', sizeof(str)); snprintf (str, sizeof(str)-1, "%s\n", drivers_get_info()); str[sizeof(str)-1] = '\0'; sock_send_string (c->sock, str); return 0; }