2005-03-19 23:16:49 +01:00
|
|
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
|
|
|
|
|
|
/*
|
2009-07-30 14:54:17 +02:00
|
|
|
* Copyright 2005-2009 by Eric House (xwords@eehouse.org). All rights
|
|
|
|
* reserved.
|
2005-03-19 23:16:49 +01:00
|
|
|
*
|
|
|
|
* 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 <unistd.h>
|
|
|
|
#include <netdb.h> /* gethostbyname */
|
|
|
|
#include <errno.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2005-03-31 04:20:50 +02:00
|
|
|
#include <iostream>
|
|
|
|
#include <sstream>
|
2005-03-19 23:16:49 +01:00
|
|
|
#include <pthread.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <sys/select.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <sys/time.h>
|
2013-08-19 17:25:06 +02:00
|
|
|
#include <glib.h>
|
2005-03-19 23:16:49 +01:00
|
|
|
|
|
|
|
#include "ctrl.h"
|
|
|
|
#include "cref.h"
|
2005-09-02 08:56:34 +02:00
|
|
|
#include "crefmgr.h"
|
2005-10-23 17:49:48 +02:00
|
|
|
#include "mlock.h"
|
2005-03-19 23:16:49 +01:00
|
|
|
#include "xwrelay_priv.h"
|
2007-11-10 06:41:49 +01:00
|
|
|
#include "configs.h"
|
2007-12-01 16:00:30 +01:00
|
|
|
#include "lstnrmgr.h"
|
2010-09-18 17:47:56 +02:00
|
|
|
#include "tpool.h"
|
2013-08-15 17:44:33 +02:00
|
|
|
#include "devmgr.h"
|
2005-03-19 23:16:49 +01:00
|
|
|
|
2005-03-25 04:24:00 +01:00
|
|
|
/* this is *only* for testing. Don't abuse!!!! */
|
2005-04-08 16:17:28 +02:00
|
|
|
extern pthread_rwlock_t gCookieMapRWLock;
|
2005-03-25 04:24:00 +01:00
|
|
|
|
2007-11-10 06:41:49 +01:00
|
|
|
/* Return of true means exit the ctrl thread */
|
2013-08-19 17:25:06 +02:00
|
|
|
typedef bool (*CmdPtr)( int socket, const char* cmd, int argc, gchar** args );
|
2005-03-31 04:20:50 +02:00
|
|
|
|
|
|
|
typedef struct FuncRec {
|
2007-11-13 04:53:10 +01:00
|
|
|
const char* name;
|
2005-03-31 04:20:50 +02:00
|
|
|
CmdPtr func;
|
|
|
|
} FuncRec;
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
vector<int> g_ctrlSocks;
|
|
|
|
pthread_mutex_t g_ctrlSocksMutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
|
|
|
|
|
2013-08-19 17:25:06 +02:00
|
|
|
static bool cmd_quit( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_print( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_devs( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
/* static bool cmd_lock( int socket, gchar** args ); */
|
|
|
|
static bool cmd_help( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_start( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_stop( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
/* static bool cmd_kill_eject( int socket, gchar** args ); */
|
|
|
|
static bool cmd_get( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_set( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_shutdown( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_rev( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_uptime( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
static bool cmd_crash( int socket, const char* cmd, int argc, gchar** args );
|
|
|
|
|
|
|
|
static void print_prompt( int socket );
|
|
|
|
|
2005-03-31 04:20:50 +02:00
|
|
|
|
2007-12-01 16:00:30 +01:00
|
|
|
static int
|
2008-05-21 05:49:11 +02:00
|
|
|
match( string* cmd, const char * const* first, int incr, int count )
|
2007-12-01 16:00:30 +01:00
|
|
|
{
|
|
|
|
int cmdlen = cmd->length();
|
|
|
|
int nFound = 0;
|
|
|
|
const char* cmdFound = NULL;
|
|
|
|
int which = -1;
|
2008-12-30 06:13:30 +01:00
|
|
|
int ii;
|
|
|
|
for ( ii = 0; (ii < count) && (nFound <= 1); ++ii ) {
|
2007-12-01 16:00:30 +01:00
|
|
|
if ( 0 == strncmp( cmd->c_str(), *first, cmdlen ) ) {
|
|
|
|
++nFound;
|
2008-12-30 06:13:30 +01:00
|
|
|
which = ii;
|
2007-12-01 16:00:30 +01:00
|
|
|
cmdFound = *first;
|
|
|
|
}
|
|
|
|
first = (char* const*)(((char*)first) + incr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( nFound == 1 ) {
|
|
|
|
cmd->assign(cmdFound);
|
|
|
|
} else {
|
|
|
|
which = -1;
|
|
|
|
}
|
|
|
|
return which;
|
|
|
|
}
|
|
|
|
|
2005-03-19 23:16:49 +01:00
|
|
|
static void
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( int sock, bool addCR, const char* what, ... )
|
2005-03-19 23:16:49 +01:00
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
|
|
|
|
va_list ap;
|
|
|
|
va_start( ap, what );
|
|
|
|
vsnprintf( buf, sizeof(buf) - 1, what, ap );
|
|
|
|
va_end(ap);
|
|
|
|
|
2005-10-14 10:26:56 +02:00
|
|
|
if ( addCR ) {
|
|
|
|
strncat( buf, "\n", sizeof(buf) );
|
|
|
|
}
|
2005-03-19 23:16:49 +01:00
|
|
|
send( sock, buf, strlen(buf), 0 );
|
|
|
|
}
|
|
|
|
|
2005-03-31 04:20:50 +02:00
|
|
|
static const FuncRec gFuncs[] = {
|
2005-10-15 18:28:26 +02:00
|
|
|
{ "?", cmd_help },
|
2007-11-13 04:53:10 +01:00
|
|
|
{ "crash", cmd_crash },
|
2013-08-15 17:44:33 +02:00
|
|
|
/* { "eject", cmd_kill_eject }, */
|
2007-11-13 04:53:10 +01:00
|
|
|
{ "get", cmd_get },
|
2005-10-15 18:28:26 +02:00
|
|
|
{ "help", cmd_help },
|
2013-08-15 17:44:33 +02:00
|
|
|
/* { "kill", cmd_kill_eject }, */
|
|
|
|
/* { "lock", cmd_lock }, */
|
2007-11-13 04:53:10 +01:00
|
|
|
{ "print", cmd_print },
|
2013-08-17 23:55:19 +02:00
|
|
|
{ "devs", cmd_devs },
|
2007-11-13 04:53:10 +01:00
|
|
|
{ "quit", cmd_quit },
|
|
|
|
{ "rev", cmd_rev },
|
|
|
|
{ "set", cmd_set },
|
|
|
|
{ "shutdown", cmd_shutdown },
|
2005-04-20 13:57:26 +02:00
|
|
|
{ "start", cmd_start },
|
|
|
|
{ "stop", cmd_stop },
|
2005-10-19 05:39:18 +02:00
|
|
|
{ "uptime", cmd_uptime },
|
2005-03-31 04:20:50 +02:00
|
|
|
};
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_quit( int socket, const char* cmd, int argc, gchar** args )
|
2005-03-19 23:16:49 +01:00
|
|
|
{
|
2005-03-31 04:20:50 +02:00
|
|
|
if ( 0 == strcmp( "help", args[1] ) ) {
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, "* %s (disconnect from ctrl port)",
|
|
|
|
args[0] );
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2005-03-31 04:20:50 +02:00
|
|
|
} else {
|
2007-11-10 06:41:49 +01:00
|
|
|
print_to_sock( socket, true, "bye bye" );
|
|
|
|
return true;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_cookies( int socket, CookieID theID )
|
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
CRefMgr* cmgr = CRefMgr::Get();
|
|
|
|
CookieMapIterator iter = cmgr->GetCookieIterator();
|
2005-04-20 13:57:26 +02:00
|
|
|
CookieID id;
|
|
|
|
for ( id = iter.Next(); id != 0; id = iter.Next() ) {
|
|
|
|
if ( theID == 0 || theID == id ) {
|
2005-09-02 08:56:34 +02:00
|
|
|
SafeCref scr( id );
|
2005-10-01 18:33:45 +02:00
|
|
|
string s;
|
|
|
|
scr.PrintCookieInfo( s );
|
2005-04-20 13:57:26 +02:00
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, s.c_str() );
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_start( int socket, const char* cmd, int argc, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, "* %s (unimplemented)", args[0] );
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_stop( int socket, const char* cmd, int argc, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, "* %s (unimplemented)", args[0] );
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
|
2013-08-15 17:44:33 +02:00
|
|
|
#if 0
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_kill_eject( int socket, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2008-12-30 06:13:30 +01:00
|
|
|
bool found = false;
|
2005-10-15 18:28:26 +02:00
|
|
|
int isKill = 0 == strcmp( args[0], "kill" );
|
2005-04-20 13:57:26 +02:00
|
|
|
|
|
|
|
if ( 0 == strcmp( args[1], "socket" ) ) {
|
|
|
|
int victim = atoi( args[2] );
|
|
|
|
if ( victim != 0 ) {
|
2010-09-18 17:47:56 +02:00
|
|
|
XWThreadPool::GetTPool()-> EnqueueKill( victim, "ctrl command" );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
2005-10-16 03:19:25 +02:00
|
|
|
} else if ( 0 == strcmp( args[1], "cref" ) ) {
|
2005-04-20 13:57:26 +02:00
|
|
|
const char* idhow = args[2];
|
2010-06-03 05:00:35 +02:00
|
|
|
for ( int indx = 3; ; ++indx ) {
|
|
|
|
const char* id = args[indx];
|
|
|
|
if ( idhow == NULL || id == NULL ) {
|
|
|
|
break;
|
|
|
|
}
|
2005-04-20 13:57:26 +02:00
|
|
|
if ( 0 == strcmp( idhow, "name" ) ) {
|
2009-07-28 07:15:26 +02:00
|
|
|
CRefMgr::Get()->Recycle( id );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
} else if ( 0 == strcmp( idhow, "id" ) ) {
|
2009-07-28 07:15:26 +02:00
|
|
|
CRefMgr::Get()->Recycle( atoi( id ) );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
}
|
2005-10-15 18:28:26 +02:00
|
|
|
} else if ( 0 == strcmp( args[1], "relay" ) ) {
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, "not yet unimplemented" );
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
|
2005-10-15 18:28:26 +02:00
|
|
|
const char* expl = isKill?
|
|
|
|
"silently remove from game"
|
|
|
|
: "remove from game with error to device";
|
2005-04-20 13:57:26 +02:00
|
|
|
if ( !found ) {
|
2007-11-13 04:53:10 +01:00
|
|
|
const char* msg =
|
2005-10-15 18:28:26 +02:00
|
|
|
"* %s socket <num> -- %s\n"
|
2010-06-03 05:00:35 +02:00
|
|
|
" %s cref name <connName>+\n"
|
|
|
|
" %s cref id <id>+"
|
2005-04-20 13:57:26 +02:00
|
|
|
;
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, msg, args[0], expl, args[0], args[0] );
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2005-10-15 18:28:26 +02:00
|
|
|
} /* cmd_kill_eject */
|
2013-08-15 17:44:33 +02:00
|
|
|
#endif
|
2005-04-20 13:57:26 +02:00
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_get( int socket, const char* cmd, int argc, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2007-12-01 16:00:30 +01:00
|
|
|
bool needsHelp = true;
|
|
|
|
|
|
|
|
string attr(args[1]);
|
2008-05-21 05:49:11 +02:00
|
|
|
const char* const attrs[] = { "help", "listeners", "loglevel" };
|
2007-12-01 16:00:30 +01:00
|
|
|
int index = match( &attr, attrs, sizeof(attrs[0]),
|
|
|
|
sizeof(attrs)/sizeof(attrs[0]));
|
|
|
|
|
|
|
|
switch( index ) {
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case 1: {
|
|
|
|
char buf[128];
|
|
|
|
int len = 0;
|
|
|
|
ListenersIter iter(&g_listeners, false);
|
|
|
|
for ( ; ; ) {
|
|
|
|
int listener = iter.next();
|
|
|
|
if ( listener == -1 ) {
|
|
|
|
break;
|
2007-11-10 06:41:49 +01:00
|
|
|
}
|
2007-12-01 16:00:30 +01:00
|
|
|
len += snprintf( &buf[len], sizeof(buf)-len, "%d,", listener );
|
|
|
|
}
|
|
|
|
print_to_sock( socket, true, "%s", buf );
|
|
|
|
needsHelp = false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2: {
|
|
|
|
RelayConfigs* rc = RelayConfigs::GetConfigs();
|
2009-03-05 14:49:01 +01:00
|
|
|
int level;
|
|
|
|
if ( NULL != rc && rc->GetValueFor( "LOGLEVEL", &level ) ) {
|
|
|
|
print_to_sock( socket, true, "loglevel=%d\n", level );
|
2007-12-01 16:00:30 +01:00
|
|
|
needsHelp = false;
|
|
|
|
} else {
|
|
|
|
logf( XW_LOGERROR, "RelayConfigs::GetConfigs() => NULL" );
|
2007-11-10 06:41:49 +01:00
|
|
|
}
|
|
|
|
}
|
2007-12-01 16:00:30 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
print_to_sock( socket, true, "unknown or ambiguous attribute: %s", attr.c_str() );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( needsHelp ) {
|
|
|
|
/* includes help */
|
|
|
|
print_to_sock( socket, false,
|
|
|
|
"* %s -- lists all attributes (unimplemented)\n"
|
|
|
|
"* %s listener\n"
|
|
|
|
"* %s loglevel\n"
|
|
|
|
, args[0], args[0], args[0] );
|
|
|
|
}
|
|
|
|
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2007-12-01 16:00:30 +01:00
|
|
|
} /* cmd_get */
|
2005-04-20 13:57:26 +02:00
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_set( int socket, const char* cmd, int argc, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2007-12-03 01:45:09 +01:00
|
|
|
const char* val = args[2];
|
2008-05-21 05:49:11 +02:00
|
|
|
const char* const attrs[] = { "help", "listeners", "loglevel" };
|
2007-12-03 01:45:09 +01:00
|
|
|
string attr(args[1]);
|
|
|
|
int index = match( &attr, attrs, sizeof(attrs[0]),
|
|
|
|
sizeof(attrs)/sizeof(attrs[0]));
|
|
|
|
|
2007-12-01 16:00:30 +01:00
|
|
|
bool needsHelp = true;
|
2007-12-03 01:45:09 +01:00
|
|
|
switch( index ) {
|
|
|
|
case 1:
|
|
|
|
if ( NULL != val && val[0] != '\0' ) {
|
|
|
|
istringstream str( val );
|
|
|
|
vector<int> sv;
|
|
|
|
while ( !str.eof() ) {
|
|
|
|
int sock;
|
|
|
|
char comma;
|
|
|
|
str >> sock >> comma;
|
|
|
|
logf( XW_LOGERROR, "%s: read %d", __func__, sock );
|
|
|
|
sv.push_back( sock );
|
|
|
|
}
|
|
|
|
g_listeners.SetAll( &sv );
|
|
|
|
needsHelp = false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
if ( NULL != val && val[0] != '\0' ) {
|
2007-11-10 06:41:49 +01:00
|
|
|
RelayConfigs* rc = RelayConfigs::GetConfigs();
|
|
|
|
if ( rc != NULL ) {
|
2009-03-05 14:49:01 +01:00
|
|
|
rc->SetValueFor( "LOGLEVEL", val );
|
2007-12-01 16:00:30 +01:00
|
|
|
needsHelp = false;
|
2007-11-10 06:41:49 +01:00
|
|
|
}
|
|
|
|
}
|
2007-12-03 01:45:09 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2007-12-01 16:00:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( needsHelp ) {
|
2007-12-03 01:45:09 +01:00
|
|
|
print_to_sock( socket, true,
|
|
|
|
"* %s listeners <n>,[<n>,..<n>,]\n"
|
|
|
|
"* %s loglevel <n>"
|
|
|
|
,args[0], args[0] );
|
2007-11-10 06:41:49 +01:00
|
|
|
}
|
|
|
|
return false;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
|
2009-02-28 17:15:59 +01:00
|
|
|
void
|
|
|
|
format_rev( char* buf, int len )
|
|
|
|
{
|
2010-06-05 16:25:24 +02:00
|
|
|
snprintf( buf, len, "git rev: %s", SVN_REV );
|
2009-02-28 17:15:59 +01:00
|
|
|
}
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_rev( int socket, const char* cmd, int argc, gchar** args )
|
2007-02-07 14:22:40 +01:00
|
|
|
{
|
|
|
|
if ( 0 == strcmp( args[1], "help" ) ) {
|
|
|
|
print_to_sock( socket, true,
|
|
|
|
"* %s -- prints svn rev number of build",
|
|
|
|
args[0] );
|
|
|
|
} else {
|
2009-02-28 17:15:59 +01:00
|
|
|
char buf[128];
|
|
|
|
format_rev( buf, sizeof(buf) );
|
|
|
|
print_to_sock( socket, true, "%s", buf );
|
2007-02-07 14:22:40 +01:00
|
|
|
}
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2007-02-07 14:22:40 +01:00
|
|
|
}
|
|
|
|
|
2009-02-28 17:15:59 +01:00
|
|
|
void
|
2009-07-31 14:56:04 +02:00
|
|
|
format_uptime( time_t seconds, char* buf, int len )
|
2009-02-28 17:15:59 +01:00
|
|
|
{
|
|
|
|
int days = seconds / (24*60*60);
|
|
|
|
seconds %= (24*60*60);
|
|
|
|
|
|
|
|
int hours = seconds / (60*60);
|
|
|
|
seconds %= (60*60);
|
|
|
|
|
|
|
|
int minutes = seconds / 60;
|
|
|
|
seconds %= 60;
|
|
|
|
|
2009-09-18 05:50:01 +02:00
|
|
|
if ( days > 0 ) {
|
|
|
|
snprintf( buf, len, "%d D %.2d:%.2d:%.2ld", days, hours,
|
|
|
|
minutes, seconds );
|
|
|
|
} else if ( hours > 0 ) {
|
|
|
|
snprintf( buf, len, "%.2d:%.2d:%.2ld", hours,
|
|
|
|
minutes, seconds );
|
|
|
|
} else if ( minutes > 0 ) {
|
|
|
|
snprintf( buf, len, "%.2d:%.2ld", minutes, seconds );
|
|
|
|
} else {
|
|
|
|
snprintf( buf, len, "%.2ld s", seconds );
|
|
|
|
}
|
2009-02-28 17:15:59 +01:00
|
|
|
}
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_uptime( int socket, const char* cmd, int argc, gchar** args )
|
2005-10-19 05:39:18 +02:00
|
|
|
{
|
|
|
|
if ( 0 == strcmp( args[1], "help" ) ) {
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true,
|
2005-10-19 05:39:18 +02:00
|
|
|
"* %s -- prints how long the relay's been running",
|
|
|
|
args[0] );
|
|
|
|
} else {
|
2009-02-28 17:15:59 +01:00
|
|
|
char buf[128];
|
2009-07-31 14:56:04 +02:00
|
|
|
format_uptime( uptime(), buf, sizeof(buf) );
|
2009-07-28 07:15:26 +02:00
|
|
|
print_to_sock( socket, true, "uptime: %s", buf );
|
2005-10-19 05:39:18 +02:00
|
|
|
}
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_crash( int socket, const char* cmd, int argc, gchar** args )
|
2007-11-10 06:41:49 +01:00
|
|
|
{
|
|
|
|
if ( 0 == strcmp( args[1], "help" ) ) {
|
|
|
|
print_to_sock( socket, true,
|
|
|
|
"* %s -- fires an assert (debug case) or divides-by-zero",
|
|
|
|
args[0] );
|
|
|
|
} else {
|
2009-08-20 14:12:43 +02:00
|
|
|
logf( XW_LOGERROR, "crashing..." );
|
2007-11-10 06:41:49 +01:00
|
|
|
assert(0);
|
2008-12-30 06:13:30 +01:00
|
|
|
int ii = 1;
|
|
|
|
while ( ii > 0 ) --ii;
|
|
|
|
return 6/ii > 0;
|
2007-11-10 06:41:49 +01:00
|
|
|
}
|
|
|
|
return false;
|
2005-10-19 05:39:18 +02:00
|
|
|
}
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_shutdown( int socket, const char* cmd, int argc, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true,
|
2005-10-15 18:28:26 +02:00
|
|
|
"* %s -- shuts down relay (exiting main) (unimplemented)",
|
|
|
|
args[0] );
|
2007-11-10 06:41:49 +01:00
|
|
|
return false;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2005-10-16 03:19:25 +02:00
|
|
|
print_cookies( int socket, const char* cookie, const char* connName )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2005-09-14 07:14:41 +02:00
|
|
|
CookieMapIterator iter = CRefMgr::Get()->GetCookieIterator();
|
|
|
|
CookieID id;
|
|
|
|
|
|
|
|
for ( id = iter.Next(); id != 0; id = iter.Next() ) {
|
|
|
|
SafeCref scr( id );
|
2010-05-01 04:49:19 +02:00
|
|
|
if ( cookie != NULL && 0 == strcasecmp( scr.Cookie(), cookie ) ) {
|
2005-10-16 03:19:25 +02:00
|
|
|
/* print this one */
|
|
|
|
} else if ( connName != NULL &&
|
|
|
|
0 == strcmp( scr.ConnName(), connName ) ) {
|
|
|
|
/* print this one */
|
|
|
|
} else {
|
|
|
|
continue;
|
2005-09-14 07:14:41 +02:00
|
|
|
}
|
2005-10-16 03:19:25 +02:00
|
|
|
string s;
|
|
|
|
scr.PrintCookieInfo( s );
|
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, s.c_str() );
|
2005-09-14 07:14:41 +02:00
|
|
|
}
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
|
2013-08-15 17:44:33 +02:00
|
|
|
#if 0
|
2005-04-20 13:57:26 +02:00
|
|
|
static void
|
|
|
|
print_socket_info( int out, int which )
|
|
|
|
{
|
|
|
|
string s;
|
2005-09-02 08:56:34 +02:00
|
|
|
CRefMgr::Get()->PrintSocketInfo( which, s );
|
2005-10-14 10:26:56 +02:00
|
|
|
print_to_sock( out, 1, s.c_str() );
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
2013-08-15 17:44:33 +02:00
|
|
|
#endif
|
2005-04-20 13:57:26 +02:00
|
|
|
|
|
|
|
static void
|
|
|
|
print_sockets( int out, int sought )
|
|
|
|
{
|
2013-08-15 17:44:33 +02:00
|
|
|
/* SocketsIterator iter = CRefMgr::Get()->MakeSocketsIterator(); */
|
|
|
|
/* int sock; */
|
|
|
|
/* while ( (sock = iter.Next()) != 0 ) { */
|
|
|
|
/* if ( sought == 0 || sought == sock ) { */
|
|
|
|
/* print_socket_info( out, sock ); */
|
|
|
|
/* } */
|
|
|
|
/* } */
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
2005-03-19 23:16:49 +01:00
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_print( int socket, const char* cmd, int argc, gchar** args )
|
2005-04-20 13:57:26 +02:00
|
|
|
{
|
2008-12-30 06:13:30 +01:00
|
|
|
bool found = false;
|
2005-10-16 03:19:25 +02:00
|
|
|
if ( 0 == strcmp( "cref", args[1] ) ) {
|
2005-04-20 13:57:26 +02:00
|
|
|
if ( 0 == strcmp( "all", args[2] ) ) {
|
|
|
|
print_cookies( socket, (CookieID)0 );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-10-16 03:19:25 +02:00
|
|
|
} else if ( 0 == strcmp( "cookie", args[2] ) ) {
|
|
|
|
print_cookies( socket, args[3], NULL );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-10-16 03:19:25 +02:00
|
|
|
} else if ( 0 == strcmp( "connName", args[2] ) ) {
|
|
|
|
print_cookies( socket, NULL, args[3] );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
} else if ( 0 == strcmp( "id", args[2] ) ) {
|
|
|
|
print_cookies( socket, atoi(args[3]) );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
} else if ( 0 == strcmp( "socket", args[1] ) ) {
|
|
|
|
if ( 0 == strcmp( "all", args[2] ) ) {
|
|
|
|
print_sockets( socket, 0 );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
} else if ( 0 == strcmp( "id", args[2] ) ) {
|
|
|
|
print_sockets( socket, atoi(args[3]) );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 13:57:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !found ) {
|
2007-11-13 04:53:10 +01:00
|
|
|
const char* str =
|
2005-10-16 03:19:25 +02:00
|
|
|
"* %s cref all\n"
|
|
|
|
" %s cref name <name>\n"
|
|
|
|
" %s cref connName <name>\n"
|
|
|
|
" %s cref id <id>\n"
|
2013-08-15 17:44:33 +02:00
|
|
|
" %s dev all -- list all known devices (by how recently connected)\n"
|
2005-10-15 18:28:26 +02:00
|
|
|
" %s socket all\n"
|
2005-10-16 03:19:25 +02:00
|
|
|
" %s socket <num> -- print info about crefs and sockets";
|
2013-08-15 17:44:33 +02:00
|
|
|
print_to_sock( socket, true, str, args[0], args[0], args[0],
|
|
|
|
args[0], args[0], args[0], args[0] );
|
2005-03-31 04:20:50 +02:00
|
|
|
}
|
2008-12-30 06:13:30 +01:00
|
|
|
return false;
|
2005-10-14 10:26:56 +02:00
|
|
|
} /* cmd_print */
|
2005-03-19 23:16:49 +01:00
|
|
|
|
2013-08-19 05:52:09 +02:00
|
|
|
/* NOTE: this will probably crash if socket is closed before the ack comes
|
|
|
|
back or times out */
|
|
|
|
static void
|
|
|
|
onAckProc( bool acked, DevIDRelay devid, uint32_t packetID, void* data )
|
|
|
|
{
|
|
|
|
int socket = (int)data;
|
|
|
|
if ( acked ) {
|
|
|
|
print_to_sock( socket, true, "got ack for packet %d from dev %d",
|
|
|
|
packetID, devid );
|
|
|
|
} else {
|
|
|
|
print_to_sock( socket, true, "NO ACK for packetID %d from dev %d",
|
|
|
|
packetID, devid );
|
|
|
|
}
|
2013-08-19 17:25:06 +02:00
|
|
|
print_prompt( socket );
|
2013-08-19 05:52:09 +02:00
|
|
|
}
|
|
|
|
|
2013-08-17 23:55:19 +02:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_devs( int socket, const char* cmd, int argc, gchar** args )
|
2013-08-17 23:55:19 +02:00
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
string result;
|
2013-08-19 17:25:06 +02:00
|
|
|
if ( 1 >= argc ) {
|
|
|
|
/* missing param; let help print */
|
|
|
|
} else if ( 0 == strcmp( "print", args[1] ) ) {
|
2013-08-17 23:55:19 +02:00
|
|
|
DevIDRelay devid = 0;
|
2013-08-19 17:25:06 +02:00
|
|
|
if ( 2 < argc ) {
|
|
|
|
devid = (DevIDRelay)strtoul( args[2], NULL, 10 );
|
2013-08-17 23:55:19 +02:00
|
|
|
}
|
|
|
|
DevMgr::Get()->printDevices( result, devid );
|
|
|
|
found = true;
|
|
|
|
} else if ( 0 == strcmp( "ping", args[1] ) ) {
|
2013-08-19 17:25:06 +02:00
|
|
|
} else if ( 0 == strcmp( "msg", args[1] ) && 3 < argc ) {
|
2013-08-17 23:55:19 +02:00
|
|
|
DevIDRelay devid = (DevIDRelay)strtoul( args[2], NULL, 10 );
|
|
|
|
const char* msg = args[3];
|
2013-08-19 05:52:09 +02:00
|
|
|
if ( post_message( devid, msg, onAckProc, (void*)socket ) ) {
|
2013-08-17 23:55:19 +02:00
|
|
|
string_printf( result, "posted message: %s\n", msg );
|
|
|
|
} else {
|
|
|
|
string_printf( result, "unable to post; does dev %d exist\n",
|
|
|
|
devid );
|
|
|
|
}
|
|
|
|
found = true;
|
2013-08-22 15:29:20 +02:00
|
|
|
} else if ( 0 == strcmp( "rm", args[1] ) && 2 < argc ) {
|
|
|
|
DevIDRelay devid = (DevIDRelay)strtoul( args[2], NULL, 10 );
|
|
|
|
if ( DevMgr::Get()->forgetDevice( devid ) ) {
|
|
|
|
string_printf( result, "dev %d removed\n", devid );
|
|
|
|
} else {
|
|
|
|
string_printf( result, "dev %d unknown\n", devid );
|
|
|
|
}
|
|
|
|
found = true;
|
2013-08-17 23:55:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( found ) {
|
|
|
|
if ( 0 < result.size() ) {
|
|
|
|
send( socket, result.c_str(), result.size(), 0 );
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const char* strs[] = {
|
|
|
|
"* %s print [<id>]\n",
|
|
|
|
" %s ping\n",
|
|
|
|
" %s msg <devid> <msg_text>\n",
|
|
|
|
" %s rm <devid>\n",
|
|
|
|
};
|
|
|
|
string help;
|
|
|
|
for ( size_t ii = 0; ii < VSIZE(strs); ++ii ) {
|
|
|
|
string_printf( help, strs[ii], args[0] );
|
|
|
|
}
|
|
|
|
send( socket, help.c_str(), help.size(), 0 );
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-08-15 17:44:33 +02:00
|
|
|
#if 0
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_lock( int socket, gchar** args )
|
2005-03-19 23:16:49 +01:00
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
CRefMgr* mgr = CRefMgr::Get();
|
2005-03-31 04:20:50 +02:00
|
|
|
if ( 0 == strcmp( "on", args[1] ) ) {
|
2005-09-02 08:56:34 +02:00
|
|
|
mgr->LockAll();
|
2005-03-31 04:20:50 +02:00
|
|
|
} else if ( 0 == strcmp( "off", args[1] ) ) {
|
2005-09-02 08:56:34 +02:00
|
|
|
mgr->UnlockAll();
|
2005-03-19 23:16:49 +01:00
|
|
|
} else {
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, true, "* %s [on|off] -- lock/unlock access mutex",
|
2005-10-15 18:28:26 +02:00
|
|
|
args[0] );
|
2005-03-19 23:16:49 +01:00
|
|
|
}
|
2005-03-31 04:20:50 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
} /* cmd_lock */
|
2013-08-15 17:44:33 +02:00
|
|
|
#endif
|
2005-03-31 04:20:50 +02:00
|
|
|
|
2007-02-07 14:22:40 +01:00
|
|
|
static bool
|
2013-08-19 17:25:06 +02:00
|
|
|
cmd_help( int socket, const char* cmd, int argc, gchar** argv )
|
2005-03-31 04:20:50 +02:00
|
|
|
{
|
2013-08-19 17:25:06 +02:00
|
|
|
if ( 1 < argc && NULL != argv[1] && 0 == strcmp( "help", argv[1] ) ) {
|
|
|
|
print_to_sock( socket, true, "* %s -- prints this", argv[0] );
|
2005-03-31 04:20:50 +02:00
|
|
|
} else {
|
|
|
|
|
2013-08-19 17:25:06 +02:00
|
|
|
gchar* help[] = { NULL, (gchar*)"help" };
|
2005-03-31 04:20:50 +02:00
|
|
|
const FuncRec* fp = gFuncs;
|
|
|
|
const FuncRec* last = fp + (sizeof(gFuncs) / sizeof(gFuncs[0]));
|
2013-08-19 17:25:06 +02:00
|
|
|
while ( fp < last ) {
|
|
|
|
help[0] = (gchar*)fp->name;
|
|
|
|
(*fp->func)( socket, (gchar*)fp->name, VSIZE(help), help );
|
2005-03-31 04:20:50 +02:00
|
|
|
++fp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-10-14 10:26:56 +02:00
|
|
|
static void
|
|
|
|
print_prompt( int socket )
|
|
|
|
{
|
2007-02-07 14:22:40 +01:00
|
|
|
print_to_sock( socket, false, "=> " );
|
2005-10-14 10:26:56 +02:00
|
|
|
}
|
|
|
|
|
2005-03-30 03:52:10 +02:00
|
|
|
static void*
|
2005-03-19 23:16:49 +01:00
|
|
|
ctrl_thread_main( void* arg )
|
|
|
|
{
|
2010-10-05 05:03:19 +02:00
|
|
|
blockSignals();
|
2007-12-01 16:00:30 +01:00
|
|
|
int sock = (int)arg;
|
2005-03-19 23:16:49 +01:00
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
{
|
|
|
|
MutexLock ml( &g_ctrlSocksMutex );
|
2007-12-01 16:00:30 +01:00
|
|
|
g_ctrlSocks.push_back( sock );
|
2005-10-23 17:49:48 +02:00
|
|
|
}
|
|
|
|
|
2013-08-19 17:25:06 +02:00
|
|
|
gint argc = 0;
|
|
|
|
gchar** args = NULL;
|
|
|
|
int index = -1;
|
2013-08-17 23:55:19 +02:00
|
|
|
string cmd;
|
|
|
|
|
2005-03-19 23:16:49 +01:00
|
|
|
for ( ; ; ) {
|
2010-06-03 05:00:35 +02:00
|
|
|
|
2007-12-01 16:00:30 +01:00
|
|
|
print_prompt( sock );
|
2005-10-14 10:26:56 +02:00
|
|
|
|
2005-03-19 23:16:49 +01:00
|
|
|
char buf[512];
|
2007-12-01 16:00:30 +01:00
|
|
|
ssize_t nGot = recv( sock, buf, sizeof(buf)-1, 0 );
|
2013-08-19 17:25:06 +02:00
|
|
|
if ( 0 >= nGot ) {
|
2005-03-19 23:16:49 +01:00
|
|
|
break;
|
2013-08-19 17:25:06 +02:00
|
|
|
} else if ( 1 == nGot ) {
|
|
|
|
assert( 0 ); /* not happening, as getting \r\n terminator */
|
|
|
|
} else if ( 2 == nGot ) {
|
|
|
|
/* user hit return; repeat prev command */
|
|
|
|
} else {
|
|
|
|
buf[nGot-2] = '\0';
|
|
|
|
if ( NULL != args ) {
|
|
|
|
g_strfreev( args );
|
2010-06-03 05:00:35 +02:00
|
|
|
}
|
2005-03-19 23:16:49 +01:00
|
|
|
|
2013-08-19 17:25:06 +02:00
|
|
|
if ( !g_shell_parse_argv( buf, &argc, &args, NULL ) ) {
|
|
|
|
assert( 0 );
|
|
|
|
} else {
|
|
|
|
cmd = args[0];
|
|
|
|
index = match( &cmd, (char*const*)&gFuncs[0].name,
|
|
|
|
sizeof(gFuncs[0]),
|
|
|
|
sizeof(gFuncs)/sizeof(gFuncs[0]) );
|
|
|
|
}
|
|
|
|
}
|
2007-12-01 16:00:30 +01:00
|
|
|
if ( index == -1 ) {
|
2008-12-30 06:13:30 +01:00
|
|
|
print_to_sock( sock, 1, "unknown or ambiguous command: \"%s\"",
|
|
|
|
cmd.c_str() );
|
2013-08-19 17:25:06 +02:00
|
|
|
gchar* args[] = { (gchar*)cmd.c_str() };
|
|
|
|
(void)cmd_help( sock, "help", 1, args );
|
|
|
|
} else if ( (*gFuncs[index].func)( sock, cmd.c_str(),
|
|
|
|
argc, args ) ) {
|
2005-03-19 23:16:49 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-10-23 17:49:48 +02:00
|
|
|
|
2007-12-01 16:00:30 +01:00
|
|
|
close ( sock );
|
2005-10-23 17:49:48 +02:00
|
|
|
|
|
|
|
MutexLock ml( &g_ctrlSocksMutex );
|
|
|
|
vector<int>::iterator iter = g_ctrlSocks.begin();
|
|
|
|
while ( iter != g_ctrlSocks.end() ) {
|
2007-12-01 16:00:30 +01:00
|
|
|
if ( *iter == sock ) {
|
2005-10-23 17:49:48 +02:00
|
|
|
g_ctrlSocks.erase(iter);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-04-08 16:17:28 +02:00
|
|
|
return NULL;
|
2005-04-20 13:57:26 +02:00
|
|
|
} /* ctrl_thread_main */
|
2005-03-30 03:52:10 +02:00
|
|
|
|
|
|
|
void
|
2005-10-23 17:49:48 +02:00
|
|
|
run_ctrl_thread( int ctrl_sock )
|
2005-03-30 03:52:10 +02:00
|
|
|
{
|
2009-08-20 14:12:43 +02:00
|
|
|
logf( XW_LOGINFO, "calling accept on socket %d", ctrl_sock );
|
2005-03-30 03:52:10 +02:00
|
|
|
|
|
|
|
sockaddr newaddr;
|
|
|
|
socklen_t siz = sizeof(newaddr);
|
2005-10-23 17:49:48 +02:00
|
|
|
int newSock = accept( ctrl_sock, &newaddr, &siz );
|
2005-10-02 18:08:42 +02:00
|
|
|
logf( XW_LOGINFO, "got one for ctrl: %d", newSock );
|
2005-03-30 03:52:10 +02:00
|
|
|
|
|
|
|
pthread_t thread;
|
|
|
|
int result = pthread_create( &thread, NULL,
|
|
|
|
ctrl_thread_main, (void*)newSock );
|
2005-10-23 17:49:48 +02:00
|
|
|
pthread_detach( thread );
|
|
|
|
|
2005-04-20 13:57:26 +02:00
|
|
|
assert( result == 0 );
|
2005-03-19 23:16:49 +01:00
|
|
|
}
|
2005-10-23 17:49:48 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
stop_ctrl_threads()
|
|
|
|
{
|
|
|
|
MutexLock ml( &g_ctrlSocksMutex );
|
|
|
|
vector<int>::iterator iter = g_ctrlSocks.begin();
|
|
|
|
while ( iter != g_ctrlSocks.end() ) {
|
|
|
|
int sock = *iter++;
|
|
|
|
print_to_sock( sock, 1, "relay going down..." );
|
|
|
|
close( sock );
|
|
|
|
}
|
|
|
|
}
|