/* -*- compile-command: "make rq"; -*- */ /* * Copyright 2010 by Eric House (xwords@eehouse.org). All rights reserved. * * 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include <string.h> #include <arpa/inet.h> #include <netdb.h> #include <sys/stat.h> #include <fcntl.h> #include <errno.h> #include <assert.h> #include "xwrelay.h" #ifndef DEFAULT_PORT # define DEFAULT_PORT 10998 #endif #ifndef DEFAULT_HOST # define DEFAULT_HOST "localhost" #endif #define MAX_CONN_NAMES 128 /* * Query: * list of all public games by language and number of players * * For debugging (localhost only, maybe) * count of all games in-play (cid!=null) */ static void usage( const char * const argv0 ) { fprintf( stderr, "usage: %s \\\n", argv0 ); fprintf( stderr, "\t[-p <port>] # (default %d) \\\n", DEFAULT_PORT ); fprintf( stderr, "\t[-a <host>] # (default: %s) \\\n", DEFAULT_HOST ); fprintf( stderr, "\t[-r] # list open public rooms \\\n" ); fprintf( stderr, "\t[-f <connName:devid> # fetch message > stdout or file \\\n" ); fprintf( stderr, "\t[-l <n>] # language for rooms " "(1=English default) \\\n" ); fprintf( stderr, "\t[-n <n>] # number of players (2 default) \\\n" ); fprintf( stderr, "\t[-o <path>]* # file to be used for -f " "(- = stdout, the default) \\\n" ); fprintf( stderr, "\t[-m <connName/devid> # list msg count \\\n" ); fprintf( stderr, "\t[-d <connName/devid/seed> # delete game \\\n" ); exit( 1 ); } int read_packet( int sock, unsigned char* buf, int buflen ) { int result = -1; ssize_t nread; unsigned short msgLen; nread = recv( sock, &msgLen, sizeof(msgLen), MSG_WAITALL ); if ( nread == sizeof(msgLen) ) { msgLen = ntohs( msgLen ); if ( msgLen <= buflen ) { nread = recv( sock, buf, msgLen, MSG_WAITALL ); if ( nread == msgLen ) { result = nread; } } } return result; } static void do_rooms( int sockfd, int lang, int nPlayers ) { unsigned char msg[] = { 0, /* protocol */ PRX_PUB_ROOMS, lang, nPlayers }; unsigned short len = htons( sizeof(msg) ); write( sockfd, &len, sizeof(len) ); write( sockfd, msg, sizeof(msg) ); fprintf( stderr, "Waiting for response....\n" ); ssize_t nRead = recv( sockfd, &len, sizeof(len), MSG_WAITALL ); assert( nRead == sizeof(len) ); len = ntohs( len ); char reply[len]; nRead = recv( sockfd, reply, len, MSG_WAITALL ); assert( nRead == len ); char* ptr = reply; unsigned short nRooms; memcpy( &nRooms, ptr, sizeof(nRooms) ); ptr += sizeof( nRooms ); nRooms = ntohs( nRooms ); int ii; char* saveptr; for ( ii = 0; ii < nRooms; ++ii ) { char* str = strtok_r( ptr, "\n", &saveptr ); fprintf( stdout, "%s\n", str ); ptr = NULL; } } static void write_connnames( int sockfd, char cmd, const char** connNames, int nConnNames ) { unsigned short len, netlen; int ii; for ( len = 0, ii = 0; ii < nConnNames; ++ii ) { len += 1 + strlen( connNames[ii] ); } unsigned char hdr[] = { 0, cmd }; unsigned short netNConnNames = htons( nConnNames ); netlen = sizeof(hdr) + sizeof( netNConnNames ) + len; netlen = htons( netlen ); write( sockfd, &netlen, sizeof(netlen) ); write( sockfd, &hdr, sizeof(hdr) ); write( sockfd, &netNConnNames, sizeof(netNConnNames) ); for ( len = 0, ii = 0; ii < nConnNames; ++ii ) { write( sockfd, connNames[ii], strlen(connNames[ii]) ); write( sockfd, "\n", 1 ); } } static void do_msgs( int sockfd, const char** connNames, int nConnNames ) { write_connnames( sockfd, PRX_HAS_MSGS, connNames, nConnNames ); fprintf( stderr, "Waiting for response....\n" ); unsigned char reply[1024]; int nRead = read_packet( sockfd, reply, sizeof(reply) ); if ( nRead > 2 ) { const unsigned char* bufp = reply; const unsigned char* const end = bufp + nRead; unsigned short count; memcpy( &count, bufp, sizeof( count ) ); bufp += sizeof( count ); count = ntohs( count ); assert( count == nConnNames ); fprintf( stderr, "got count: %d\n", count ); int ii; for ( ii = 0; ii < nConnNames && bufp < end; ++ii ) { memcpy( &count, bufp, sizeof( count ) ); bufp += sizeof( count ); count = ntohs( count ); fprintf( stdout, "%s -- %d\n", connNames[ii], count ); } } } /* do_msgs */ static void do_fetch( int sockfd, const char** connNames, int nConnNames, const char** pipes, int nPipes ) { write_connnames( sockfd, PRX_GET_MSGS, connNames, nConnNames ); fprintf( stderr, "Waiting for response....\n" ); unsigned char reply[1024]; int nRead = read_packet( sockfd, reply, sizeof(reply) ); if ( nRead > 2 ) { const unsigned char* bufp = reply; const unsigned char* const end = bufp + nRead; unsigned short count; memcpy( &count, bufp, sizeof( count ) ); bufp += sizeof( count ); count = ntohs( count ); assert( count <= nConnNames ); fprintf( stderr, "got count: %d\n", count ); /* Now we have an array of <countPerDev> <len><len bytes> pairs. Just write em as long as it makes sense. countPerDev makes no sense as other than 1 unless the UI is changed so I don't have to write to STDOUT -- e.g. by passing in named pipes to correspond to each deviceid provided */ for ( int ii = 0; ii < count && bufp < end; ++ii ) { int fd = STDOUT_FILENO; unsigned short countPerDev; memcpy( &countPerDev, bufp, sizeof( countPerDev ) ); bufp += sizeof( countPerDev ); countPerDev = ntohs( countPerDev ); if ( ii < nPipes && 0 != strcmp( pipes[ii], "-" ) ) { fd = open( pipes[ii], O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR ); if ( fd < 0 ) { fprintf( stderr, "open(%s) failed: %s\n", pipes[ii], strerror(errno) ); exit( 1 ); } } while ( bufp < end && countPerDev-- > 0 ) { unsigned short len; memcpy( &len, bufp, sizeof( len ) ); len = ntohs( len ) + sizeof( len ); if ( bufp + len > end ) { break; } fprintf( stderr, "writing %d bytes to fd %d\n", len, fd ); write( fd, bufp, len ); bufp += len; } if ( fd != STDOUT_FILENO ) { close( fd ); } } if ( bufp != end ) { fprintf( stderr, "error: message not internally as expected\n" ); } } } static void do_deletes( int sockfd, const char** connNames, int nConnNames ) { int ii; char buf[4096]; int nused = 0; for ( ii = 0; ii < nConnNames; ++ii ) { char tmp[32]; strcpy( tmp, connNames[ii] ); char* seedp = strrchr( tmp, '/' ); assert( !!seedp ); *(seedp++) = '\0'; /* skip */ unsigned int seed; sscanf( seedp, "%X", &seed ); short netshort = htons( (short)seed ); memcpy( &buf[nused], &netshort, sizeof(netshort) ); nused += sizeof(netshort); nused += sprintf( &buf[nused], "%s", tmp ); buf[nused++] = '\n'; } unsigned char hdr[] = { 0, PRX_DEVICE_GONE }; /* total len */ short num = htons( sizeof(hdr) + 2 + nused ); write( sockfd, &num, sizeof(num) ); /* header */ write( sockfd, hdr, sizeof(hdr) ); /* count */ num = htons( nConnNames ); write( sockfd, &num, sizeof(num) ); /* buffer */ write( sockfd, buf, nused ); unsigned char reply[2]; int nRead = read_packet( sockfd, reply, sizeof(reply) ); if ( nRead != 0 ) { fprintf( stderr, "%s: nRead=%d; errno=%d (%s)\n", __func__, nRead, errno, strerror(errno) ); } } /* do_deletes */ int main( int argc, char * const argv[] ) { int port = DEFAULT_PORT; int lang = 1; int nPlayers = 2; int doWhat = 0; const int doRooms = 1; const int doMgs = 2; const int doDeletes = 4; const int doFetch = 8; const char* host = DEFAULT_HOST; char const* connNames[MAX_CONN_NAMES]; int nConnNames = 0; const char* pipes[MAX_CONN_NAMES]; int nPipes = 0; for ( ; ; ) { int opt = getopt( argc, argv, "a:d:f:p:rl:n:m:o:" ); if ( opt < 0 ) { break; } switch ( opt ) { case 'a': host = optarg; break; case 'd': assert( nConnNames < MAX_CONN_NAMES - 1 ); connNames[nConnNames++] = optarg; doWhat |= doDeletes; break; case 'f': connNames[nConnNames++] = optarg; doWhat |= doFetch; break; case 'l': lang = atoi(optarg); break; case 'm': assert( nConnNames < MAX_CONN_NAMES - 1 ); connNames[nConnNames++] = optarg; doWhat |= doMgs; break; case 'n': nPlayers = atoi(optarg); break; case 'o': assert( nPipes < MAX_CONN_NAMES - 1 ); pipes[nPipes++] = optarg; break; case 'p': port = atoi(optarg); break; case 'r': doWhat |= doRooms; break; default: usage( argv[0] ); break; } } fprintf( stderr, "got port %d, host %s\n", port, host ); int sockfd = socket( AF_INET, SOCK_STREAM, 0 ); struct sockaddr_in to_sock; memset( &to_sock, 0, sizeof(to_sock) ); to_sock.sin_family = AF_INET; to_sock.sin_port = htons( port ); struct hostent* hostip; hostip = gethostbyname( host ); memcpy( &(to_sock.sin_addr.s_addr), hostip->h_addr_list[0], sizeof(hostip->h_addr_list[0] ) ); if ( 0 != connect( sockfd, (const struct sockaddr*)&to_sock, sizeof(to_sock) ) ) { fprintf( stderr, "connect failed: %d (%s)\n", errno, strerror(errno) ); exit( 1 ); } switch ( doWhat ) { case 0: fprintf( stderr, "no command given\n" ); usage( argv[0] ); break; case doRooms: do_rooms( sockfd, lang, nPlayers ); break; case doMgs: do_msgs( sockfd, connNames, nConnNames ); break; case doDeletes: do_deletes( sockfd, connNames, nConnNames ); break; case doFetch: do_fetch( sockfd, connNames, nConnNames, pipes, nPipes ); break; default: fprintf( stderr, "conflicting options given\n" ); usage( argv[0] ); } close( sockfd ); return 0; }