mirror of
https://github.com/gamedig/node-gamedig.git
synced 2024-11-18 17:50:37 +01:00
285 lines
5.9 KiB
Text
285 lines
5.9 KiB
Text
|
LICENSE: The Artistic License 2.0
|
||
|
|
||
|
/*
|
||
|
* qstat.h
|
||
|
* by Steve Jankowski
|
||
|
* steve@qstat.org
|
||
|
* http://www.qstat.org
|
||
|
*
|
||
|
* Copyright 1996,1997,1998,1999,2000,2001,2002 by Steve Jankowski
|
||
|
*/
|
||
|
|
||
|
char cube2_serverstatus[3] = {'\x80', '\x10', '\x27'};
|
||
|
|
||
|
|
||
|
{
|
||
|
/* Cube 2/Sauerbraten/Blood Frontier */
|
||
|
CUBE2_SERVER, /* id */
|
||
|
"CUBE2", /* type_prefix */
|
||
|
"cube2", /* type_string */
|
||
|
"-cubes", /* type_option */
|
||
|
"Sauerbraten", /* game_name */
|
||
|
0, /* master */
|
||
|
CUBE2_DEFAULT_PORT, /* default_port */
|
||
|
1, /* port_offset */
|
||
|
0, /* flags */
|
||
|
"", /* game_rule */
|
||
|
"CUBE2", /* template_var */
|
||
|
cube2_serverstatus, /* status_packet */
|
||
|
sizeof(cube2_serverstatus), /* status_len */
|
||
|
NULL, /* player_packet */
|
||
|
0, /* player_len */
|
||
|
NULL, /* rule_packet */
|
||
|
0, /* rule_len */
|
||
|
NULL, /* master_packet */
|
||
|
0, /* master_len */
|
||
|
NULL, /* master_protocol */
|
||
|
NULL, /* master_query */
|
||
|
NULL, /* display_player_func */
|
||
|
display_server_rules, /* display_rule_func */
|
||
|
NULL, /* display_raw_player_func */
|
||
|
raw_display_server_rules, /* display_raw_rule_func */
|
||
|
NULL, /* display_xml_player_func */
|
||
|
xml_display_server_rules, /* display_xml_rule_func */
|
||
|
send_cube2_request_packet, /* status_query_func */
|
||
|
NULL, /* rule_query_func */
|
||
|
NULL, /* player_query_func */
|
||
|
deal_with_cube2_packet, /* packet_func */
|
||
|
},
|
||
|
|
||
|
|
||
|
|
||
|
/*
|
||
|
* qstat 2.12
|
||
|
* by Steve Jankowski
|
||
|
*
|
||
|
* Cube 2 / Sauerbraten protocol
|
||
|
* Copyright 2011 NoisyB
|
||
|
*
|
||
|
* Licensed under the Artistic License, see LICENSE.txt for license terms
|
||
|
*/
|
||
|
#include <string.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
#include "debug.h"
|
||
|
#include "qstat.h"
|
||
|
#include "packet_manip.h"
|
||
|
|
||
|
struct offset
|
||
|
{
|
||
|
unsigned char *d;
|
||
|
int pos;
|
||
|
int len;
|
||
|
};
|
||
|
|
||
|
|
||
|
//#define SB_MASTER_SERVER "http://sauerbraten.org/masterserver/retrieve.do?item=list"
|
||
|
#define SB_PROTOCOL 258
|
||
|
#define MIN(a,b) ((a)<(b)?(a):(b))
|
||
|
#define MAX_ATTR 255
|
||
|
#define MAX_STRING 1024
|
||
|
|
||
|
|
||
|
static int
|
||
|
getint (struct offset * d)
|
||
|
{
|
||
|
int val = 0;
|
||
|
|
||
|
if ( d->pos >= d->len )
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
val = d->d[d->pos++] & 0xff; // 8 bit value
|
||
|
|
||
|
// except...
|
||
|
if ( val == 0x80 && d->pos < d->len - 2 ) // 16 bit value
|
||
|
{
|
||
|
val = (d->d[d->pos++] & 0xff);
|
||
|
val |= (d->d[d->pos++] & 0xff) << 8;
|
||
|
}
|
||
|
else if ( val == 0x81 && d->pos < d->len - 4 ) // 32 bit value
|
||
|
{
|
||
|
val = (d->d[d->pos++] & 0xff);
|
||
|
val |= (d->d[d->pos++] & 0xff) << 8;
|
||
|
val |= (d->d[d->pos++] & 0xff) << 16;
|
||
|
val |= (d->d[d->pos++] & 0xff) << 24;
|
||
|
}
|
||
|
|
||
|
return val;
|
||
|
}
|
||
|
|
||
|
|
||
|
static char * getstr( char *dest, int dest_len, struct offset *d )
|
||
|
{
|
||
|
int len = 0;
|
||
|
|
||
|
if (d->pos >= d->len)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
len = MIN( dest_len, d->len - d->pos );
|
||
|
strncpy( dest, (const char *) d->d + d->pos, len )[len - 1] = 0;
|
||
|
d->pos += strlen (dest) + 1;
|
||
|
|
||
|
return dest;
|
||
|
}
|
||
|
|
||
|
|
||
|
static char* sb_getversion_s (int n)
|
||
|
{
|
||
|
static char *version_s[] =
|
||
|
{
|
||
|
"Justice",
|
||
|
"CTF",
|
||
|
"Assassin",
|
||
|
"Summer",
|
||
|
"Spring",
|
||
|
"Gui",
|
||
|
"Water",
|
||
|
"Normalmap",
|
||
|
"Sp",
|
||
|
"Occlusion",
|
||
|
"Shader",
|
||
|
"Physics",
|
||
|
"Mp",
|
||
|
"",
|
||
|
"Agc",
|
||
|
"Quakecon",
|
||
|
"Independence"
|
||
|
};
|
||
|
|
||
|
n = SB_PROTOCOL - n;
|
||
|
if (n >= 0 && (size_t) n < sizeof(version_s) / sizeof(version_s[0]))
|
||
|
{
|
||
|
return version_s[n];
|
||
|
}
|
||
|
return "unknown";
|
||
|
}
|
||
|
|
||
|
|
||
|
static char* sb_getmode_s(int n)
|
||
|
{
|
||
|
static char *mode_s[] =
|
||
|
{
|
||
|
"slowmo SP",
|
||
|
"slowmo DMSP",
|
||
|
"demo",
|
||
|
"SP",
|
||
|
"DMSP",
|
||
|
"ffa/default",
|
||
|
"coopedit",
|
||
|
"ffa/duel",
|
||
|
"teamplay",
|
||
|
"instagib",
|
||
|
"instagib team",
|
||
|
"efficiency",
|
||
|
"efficiency team",
|
||
|
"insta arena",
|
||
|
"insta clan arena",
|
||
|
"tactics arena",
|
||
|
"tactics clan arena",
|
||
|
"capture",
|
||
|
"insta capture",
|
||
|
"regen capture",
|
||
|
"assassin",
|
||
|
"insta assassin",
|
||
|
"ctf",
|
||
|
"insta ctf"
|
||
|
};
|
||
|
|
||
|
n += 6;
|
||
|
if (n >= 0 && (size_t) n < sizeof(mode_s) / sizeof(mode_s[0]))
|
||
|
{
|
||
|
return mode_s[n];
|
||
|
}
|
||
|
return "unknown";
|
||
|
}
|
||
|
|
||
|
|
||
|
query_status_t send_cube2_request_packet( struct qserver *server )
|
||
|
{
|
||
|
return send_packet( server, server->type->status_packet, server->type->status_len );
|
||
|
}
|
||
|
|
||
|
|
||
|
query_status_t deal_with_cube2_packet( struct qserver *server, char *rawpkt, int pktlen )
|
||
|
{
|
||
|
// skip unimplemented ack, crc, etc
|
||
|
int i;
|
||
|
int numattr;
|
||
|
int attr[MAX_ATTR];
|
||
|
char buf[MAX_STRING];
|
||
|
enum {
|
||
|
MM_OPEN = 0,
|
||
|
MM_VETO,
|
||
|
MM_LOCKED,
|
||
|
MM_PRIVATE
|
||
|
};
|
||
|
struct offset d;
|
||
|
d.d = (unsigned char *) rawpkt;
|
||
|
d.pos = 0;
|
||
|
d.len = pktlen;
|
||
|
|
||
|
server->ping_total += time_delta( &packet_recv_time, &server->packet_time1 );
|
||
|
getint( &d ); // we have the ping already
|
||
|
server->num_players = getint( &d );
|
||
|
numattr = getint( &d );
|
||
|
for ( i = 0; i < numattr && i < MAX_ATTR; i++ )
|
||
|
{
|
||
|
attr[i] = getint (&d);
|
||
|
}
|
||
|
|
||
|
server->protocol_version = attr[0];
|
||
|
|
||
|
sprintf( buf, "%d %s", attr[0], sb_getversion_s (attr[0]) );
|
||
|
add_rule( server, "version", buf, NO_FLAGS );
|
||
|
|
||
|
sprintf( buf, "%d %s", attr[1], sb_getmode_s (attr[1]) );
|
||
|
add_rule( server, "mode", buf, NO_FLAGS );
|
||
|
|
||
|
sprintf( buf, "%d", attr[2] );
|
||
|
add_rule( server, "seconds_left", buf, NO_FLAGS );
|
||
|
|
||
|
server->max_players = attr[3];
|
||
|
|
||
|
switch ( attr[5] )
|
||
|
{
|
||
|
case MM_OPEN:
|
||
|
sprintf( buf, "%d open", attr[5] );
|
||
|
break;
|
||
|
case MM_VETO:
|
||
|
sprintf( buf, "%d veto", attr[5] );
|
||
|
break;
|
||
|
case MM_LOCKED:
|
||
|
sprintf( buf, "%d locked", attr[5] );
|
||
|
break;
|
||
|
case MM_PRIVATE:
|
||
|
sprintf( buf, "%d private", attr[5] );
|
||
|
break;
|
||
|
default:
|
||
|
sprintf( buf, "%d unknown", attr[5] );
|
||
|
}
|
||
|
add_rule( server, "mm", buf, NO_FLAGS);
|
||
|
|
||
|
for ( i = 0; i < numattr && i < MAX_ATTR; i++ )
|
||
|
{
|
||
|
char buf2[MAX_STRING];
|
||
|
sprintf( buf, "attr%d", i );
|
||
|
sprintf( buf2, "%d", attr[i] );
|
||
|
add_rule( server, buf, buf2, NO_FLAGS );
|
||
|
}
|
||
|
|
||
|
getstr( buf, MAX_STRING, &d );
|
||
|
server->map_name = strdup(buf);
|
||
|
getstr( buf, MAX_STRING, &d );
|
||
|
server->server_name = strdup(buf);
|
||
|
|
||
|
return DONE_FORCE;
|
||
|
}
|
||
|
|
||
|
|
||
|
|