2009-02-28 17:15:59 +01:00
|
|
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2009 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef DO_HTTP
|
|
|
|
|
|
|
|
#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>
|
|
|
|
#include <iostream>
|
|
|
|
#include <sstream>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <sys/select.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
|
|
|
#include "ctrl.h"
|
|
|
|
#include "cref.h"
|
|
|
|
#include "crefmgr.h"
|
|
|
|
#include "mlock.h"
|
|
|
|
#include "xwrelay_priv.h"
|
|
|
|
#include "configs.h"
|
|
|
|
#include "lstnrmgr.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* http://www.jbox.dk/sanos/webserver.htm has code for a trivial web server. Good example.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
send_header( FILE* fil, const char* title )
|
|
|
|
{
|
|
|
|
fprintf( fil, "HTTP/1.0 %d %s\r\n", 200, title );
|
|
|
|
fprintf( fil, "Server: xwrelay\r\n" );
|
|
|
|
fprintf( fil, "Content-Type: text/html\r\n" );
|
|
|
|
fprintf( fil, "Connection: close\r\n");
|
|
|
|
|
|
|
|
fprintf( fil, "\r\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
send_meta( FILE* fil )
|
|
|
|
{
|
|
|
|
FILE* css;
|
2009-03-02 06:29:06 +01:00
|
|
|
RelayConfigs* cfg = RelayConfigs::GetConfigs();
|
2009-07-30 14:49:21 +02:00
|
|
|
char pathbuf[256] = { '\0' };
|
2009-03-02 06:29:06 +01:00
|
|
|
|
2009-02-28 17:15:59 +01:00
|
|
|
fprintf( fil, "<head>" );
|
2009-03-02 06:29:06 +01:00
|
|
|
|
|
|
|
if ( !!cfg ) {
|
|
|
|
int refreshSecs;
|
|
|
|
if ( cfg->GetValueFor( "WWW_REFRESH_SECS", &refreshSecs ) ) {
|
|
|
|
fprintf( fil, "<meta http-equiv=\"refresh\" content=\"%d\" />",
|
|
|
|
refreshSecs );
|
|
|
|
}
|
2009-07-30 14:49:21 +02:00
|
|
|
|
|
|
|
(void)cfg->GetValueFor( "WWW_CSS_PATH", pathbuf, sizeof(pathbuf) );
|
2009-03-02 06:29:06 +01:00
|
|
|
}
|
2009-02-28 17:15:59 +01:00
|
|
|
|
2009-07-30 14:49:21 +02:00
|
|
|
if ( pathbuf[0] ) {
|
|
|
|
css = fopen( pathbuf, "r" );
|
|
|
|
if ( NULL != css ) {
|
|
|
|
for ( ; ; ) {
|
|
|
|
char buf[256];
|
|
|
|
size_t nread = fread( buf, 1, sizeof(buf), css );
|
|
|
|
if ( nread <= 0 ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
(void) fwrite( buf, 1, nread, fil );
|
2009-02-28 17:15:59 +01:00
|
|
|
}
|
2009-07-30 14:49:21 +02:00
|
|
|
fclose( css );
|
2009-02-28 17:15:59 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf( fil, "</head>" );
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
printTail( FILE* fil )
|
|
|
|
{
|
|
|
|
char buf[128];
|
|
|
|
|
|
|
|
/* print version and uptime */
|
|
|
|
fprintf( fil, "<div class=\"header\">Relay version</div>" );
|
|
|
|
format_rev( buf, sizeof(buf) );
|
|
|
|
fprintf( fil, "<p>%s</p>", buf );
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
printCrefs( FILE* fil )
|
|
|
|
{
|
|
|
|
fprintf( fil, "<div class=\"header\">Connections</div>" );
|
|
|
|
fprintf( fil, "<table><tr>" );
|
|
|
|
fprintf( fil,
|
|
|
|
"<th>Cookie</th>"
|
|
|
|
"<th>ConnName</th>"
|
|
|
|
"<th>Cookie ID</th>"
|
|
|
|
"<th>Total sent</th>"
|
|
|
|
"<th>Players</th>"
|
|
|
|
"<th>Players Here</th>"
|
|
|
|
"<th>State</th>"
|
2009-07-28 07:41:15 +02:00
|
|
|
"<th>Secs conn'd</th>"
|
2009-02-28 20:35:32 +01:00
|
|
|
"<th>Host IDs</th>"
|
2009-07-28 07:15:26 +02:00
|
|
|
"<th>Host IPs</th>"
|
2009-07-29 06:25:21 +02:00
|
|
|
"\n"
|
2009-02-28 17:15:59 +01:00
|
|
|
);
|
|
|
|
fprintf( fil, "</tr>\n" );
|
|
|
|
|
|
|
|
CRefMgr* cmgr = CRefMgr::Get();
|
|
|
|
CookieMapIterator iter = cmgr->GetCookieIterator();
|
|
|
|
CookieID id;
|
2009-07-28 07:41:15 +02:00
|
|
|
time_t curTime = uptime();
|
2009-02-28 17:15:59 +01:00
|
|
|
for ( id = iter.Next(); id != 0; id = iter.Next() ) {
|
2009-07-28 07:15:26 +02:00
|
|
|
string hosts, addrs;
|
|
|
|
SafeCref scr( id, true );
|
|
|
|
if ( scr.IsValid() ) {
|
|
|
|
scr.GetHostsConnected( &hosts, &addrs );
|
|
|
|
|
|
|
|
fprintf( fil, "<tr>"
|
|
|
|
"<td>%s</td>" /* name */
|
|
|
|
"<td>%s</td>" /* conn name */
|
|
|
|
"<td>%d</td>" /* cookie id */
|
|
|
|
"<td>%d</td>" /* total sent */
|
|
|
|
"<td>%d</td>" /* players */
|
|
|
|
"<td>%d</td>" /* players here */
|
|
|
|
"<td>%s</td>" /* State */
|
2009-07-28 07:41:15 +02:00
|
|
|
"<td>%ld</td>" /* uptime */
|
2009-07-28 07:15:26 +02:00
|
|
|
"<td>%s</td>" /* Hosts */
|
|
|
|
"<td>%s</td>" /* Ip addrs */
|
2009-07-29 06:25:21 +02:00
|
|
|
"</tr>\n",
|
2009-07-28 07:15:26 +02:00
|
|
|
scr.Cookie(), scr.ConnName(), scr.GetCookieID(),
|
|
|
|
scr.GetTotalSent(), scr.GetPlayersTotal(),
|
|
|
|
scr.GetPlayersHere(), scr.StateString(),
|
2009-07-28 07:41:15 +02:00
|
|
|
curTime - scr.GetStartTime(),
|
2009-07-28 07:15:26 +02:00
|
|
|
hosts.c_str(), addrs.c_str()
|
|
|
|
);
|
|
|
|
}
|
2009-02-28 17:15:59 +01:00
|
|
|
}
|
|
|
|
fprintf( fil, "</table>\n" );
|
|
|
|
}
|
|
|
|
|
2009-07-06 03:50:51 +02:00
|
|
|
static void
|
2009-07-13 05:01:22 +02:00
|
|
|
printStats( FILE* fil )
|
2009-07-06 03:50:51 +02:00
|
|
|
{
|
|
|
|
CRefMgr* cmgr = CRefMgr::Get();
|
|
|
|
int nGames = cmgr->GetNumGamesSeen();
|
2009-07-13 05:01:22 +02:00
|
|
|
int siz = cmgr->GetSize();
|
2009-07-31 14:56:04 +02:00
|
|
|
char uptime1[64];
|
|
|
|
char uptime2[64];
|
|
|
|
format_uptime( uptime(), uptime1, sizeof(uptime1) );
|
|
|
|
format_uptime( cmgr->uptime(), uptime2, sizeof(uptime2) );
|
2009-07-13 05:01:22 +02:00
|
|
|
fprintf( fil, "<div class=\"header\">Stats</div>" );
|
|
|
|
fprintf( fil, "<table>" );
|
2009-07-31 14:56:04 +02:00
|
|
|
fprintf( fil, "<tr>"
|
|
|
|
"<th>Uptime</th><th>Spawns</th><th>Spawn Utime</th>"
|
|
|
|
"<th>Games played</th><th>Games in play</th></tr>" );
|
|
|
|
fprintf( fil, "<tr><td>%s</td><td>%d</td>"
|
|
|
|
"<td>%s</td><td>%d</td><td>%d</td></tr>\n",
|
|
|
|
uptime1, GetNSpawns(), uptime2, nGames, siz );
|
2009-07-13 05:01:22 +02:00
|
|
|
fprintf( fil, "</table>" );
|
2009-07-06 03:50:51 +02:00
|
|
|
}
|
|
|
|
|
2009-02-28 17:15:59 +01:00
|
|
|
static void*
|
|
|
|
http_thread_main( void* arg )
|
|
|
|
{
|
|
|
|
int sock = (int)arg;
|
|
|
|
char buf[512];
|
|
|
|
ssize_t totalRead = 0;
|
|
|
|
|
|
|
|
while ( totalRead <= 4 ) { /* have we read enough for GET? */
|
|
|
|
ssize_t nread = read( sock, buf+totalRead, sizeof(buf)-1-totalRead );
|
|
|
|
if ( nread == 0 ) {
|
|
|
|
break;
|
|
|
|
} else if ( nread > 0 ) {
|
|
|
|
buf[nread] = '\0';
|
|
|
|
}
|
|
|
|
totalRead += nread;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( 0 == strncasecmp( "GET ", buf, 3 ) ) {
|
|
|
|
FILE* fil = fdopen( sock, "r+" );
|
|
|
|
fseek( fil, 0, SEEK_CUR ); // reverse stream
|
|
|
|
|
|
|
|
send_header( fil, "status page" );
|
|
|
|
fprintf( fil, "<html>" );
|
|
|
|
send_meta( fil );
|
|
|
|
fprintf( fil, "<body><div class=\"main\">" );
|
|
|
|
|
2009-07-13 05:01:22 +02:00
|
|
|
printStats( fil );
|
2009-07-06 03:50:51 +02:00
|
|
|
|
2009-02-28 17:15:59 +01:00
|
|
|
printCrefs( fil );
|
|
|
|
|
|
|
|
printTail( fil );
|
|
|
|
|
|
|
|
fprintf( fil, "</div></body></html>" );
|
|
|
|
|
|
|
|
fclose( fil );
|
|
|
|
} else {
|
|
|
|
logf( XW_LOGINFO, "NOT a GET" );
|
|
|
|
}
|
|
|
|
close( sock );
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
} /* http_thread_main */
|
|
|
|
|
|
|
|
void
|
|
|
|
run_http_thread( int http_sock )
|
|
|
|
{
|
|
|
|
sockaddr newaddr;
|
|
|
|
socklen_t siz = sizeof(newaddr);
|
|
|
|
int newSock = accept( http_sock, &newaddr, &siz );
|
|
|
|
|
|
|
|
pthread_t thread;
|
|
|
|
int result = pthread_create( &thread, NULL,
|
|
|
|
http_thread_main, (void*)newSock );
|
|
|
|
if ( 0 == result ) {
|
|
|
|
pthread_detach( thread );
|
|
|
|
} else {
|
|
|
|
logf( XW_LOGERROR, "%s: pthread_create failed", __func__ );
|
|
|
|
}
|
|
|
|
} /* run_http_thread */
|
|
|
|
|
|
|
|
#endif /* DO_HTTP */
|