/* -*- 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 #include #include #include #include #include #include #include #include #include #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 ] # (default %d)\\\n", DEFAULT_PORT ); fprintf( stderr, "\t[-a ] # (default: %s)\\\n", DEFAULT_HOST ); fprintf( stderr, "\t -r # list open public rooms\\\n" ); fprintf( stderr, "\t[-l ] # language for rooms " "(1=English default)\\\n" ); fprintf( stderr, "\t[-n ] # number of players (2 default)\\\n" ); fprintf( stderr, "\t[-m # list msg count\\\n" ); fprintf( stderr, "\t[-d # 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 do_msgs( int sockfd, 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, PRX_HAS_MSGS }; 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 ); } 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_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; bool doRooms = false; bool doMgs = false; bool doDeletes = false; const char* host = DEFAULT_HOST; char const* connNames[MAX_CONN_NAMES]; int nConnNames = 0; for ( ; ; ) { int opt = getopt( argc, argv, "a:d:p:rl:n:m:" ); if ( opt < 0 ) { break; } switch ( opt ) { case 'a': host = optarg; break; case 'd': if ( doMgs ) { fprintf( stderr, "can't mix -d and -m\n" ); usage( argv[0] ); } assert( nConnNames < MAX_CONN_NAMES - 1 ); connNames[nConnNames++] = optarg; doDeletes = true; break; case 'l': lang = atoi(optarg); break; case 'm': if ( doDeletes ) { fprintf( stderr, "can't mix -d and -m\n" ); usage( argv[0] ); } assert( nConnNames < MAX_CONN_NAMES - 1 ); connNames[nConnNames++] = optarg; doMgs = true; break; case 'n': nPlayers = atoi(optarg); break; case 'p': port = atoi(optarg); break; case 'r': doRooms = true; 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 ); } if ( doRooms ) { do_rooms( sockfd, lang, nPlayers ); } if ( doMgs ) { do_msgs( sockfd, connNames, nConnNames ); } if ( doDeletes ) { do_deletes( sockfd, connNames, nConnNames ); } close( sockfd ); return 0; }