forked from bmillwood/tremulous-master
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathconfig.py
537 lines (483 loc) · 21.8 KB
/
config.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
# config.py
# Copyright (c) Ben Millwood 2009
# Copyright (c) Darren Salt 2012
# This file is part of the Tremulous Master server.
'''Configuration for the Tremulous Master
This module provides the MasterConfig class, and the ConfigError exception.
The latter is raised when the former fails to initialise for some reason.
The former, after its parse() method is called, provides the following
instance variables:
VERSION:
a string giving the name and version of the master
CHALLENGE_LENGTH:
the length of a getinfo challenge sent to servers
CHALLENGE_TIMEOUT:
in seconds, the time a server has after being challenged to respond
SERVER_TIMEOUT:
a server that has not sent a heartbeat in this period will be forgotten
GSR_MAXSERVERS:
the maximum number of server addresses to be sent in a single
getservers[Ext]Response packet (the client will only accept so many)
FEATURED_FILE, IGNORE_FILE, MOTD_FILE:
These are mostly for internal use, but contain the file names from
which the featured servers, address blacklist, and motd are read.
game_name:
Contains the game name.
game_id:
Contains the game identifier string used by clients.
ipv4, ipv6:
These variables specify which interfaces the client will use: at least
one will be true, or ConfigError will be raised.
listen_addr, listen6_addr, port, challengeport:
The master needs a socket for incoming connections, for each of IPv4
and IPv6 that it is asked to use (currently it assumes the same port on
each). These settings tell it to what address and port it should bind.
It uses a separate port for outgoing challenges because some routers
will redirect a 'response' to the heartbeat to the correct host even if
they are not configured to do so for client requests. The separate port
defeats this connection tracking and we only get a response if ports
are set up correctly.
inPort defaults according to game; outPort defaults to inPort + 1
max_servers:
This defaults to unlimited but if someone finds a way to flood the
server list it could serve as a measure to prevent excessive RAM usage.
featured_servers:
This is a dict of lists - each key is a label, and its value is a list
of addresses. They will be sent in a separate response packet, headered
with the label, so that the client can display them specially.
and some useful functions:
log(level, arg[, arg...], sep = ' '):
level may be one of LOG_ERROR, LOG_PRINT, LOG_VERBOSE, or LOG_DEBUG:
if the user's chosen verbosity level is less, the message will not be
printed. All the subsequent arguments will be str()'d and printed,
preceded by a timestamp and joined by the string given in the keyword
argument `sep' (default ' ')
getmotd():
Simply reads the motd file and returns the result.
ignore(addr):
Reads the IGNORE_FILE and returns True if the given IP address is
blacklisted by its contents.
'''
# Required imports
from errno import ENOENT, EIO
from optparse import OptionParser, Values
from sys import argv, exit, stdout
from time import strftime
# Local imports
from utils import inet_pton, valid_addr, stringtosockaddr
# Optional imports
# I named these variables in line with the standard library's has_ipv6.
# Surely that should be have_ipv6?
has_chroot, has_setuid, has_setgid, has_setgroups = True, True, True, True
try:
from os import chdir, chroot
except ImportError:
has_chroot = False
try:
from pwd import getpwnam, getpwuid, getpwall
from os import setuid, getuid
except ImportError:
has_setuid = False
try:
from os import setgid, getgid
except ImportError:
has_setgid = False
try:
from os import setgroups, getgroups
except ImportError:
has_setgroups = False
# I don't have a non-IPv6 computer, so I'm not sure how this works
try:
from socket import has_ipv6
except ImportError:
has_ipv6 = False
( # Log levels
LOG_ALWAYS,
LOG_ERROR,
LOG_PRINT,
LOG_VERBOSE,
LOG_DEBUG,
LOG_LEVELS
) = range(6)
# and their names
loglevels = ['ALWAYS', 'ERROR', 'PRINT', 'VERBOSE', 'DEBUG']
def concat(*args, **kwargs):
# docstring TODO
try:
sep = kwargs['sep']
del kwargs['sep']
except KeyError:
sep = ' '
if kwargs:
raise TypeError('unexpected keyword arguments: ' +
str(list(kwargs.keys())))
return sep.join(map(str, args))
class ConcatError(Exception):
# docstring TODO
def __init__(self, *args, **kwargs):
Exception.__init__(self, concat(*args, **kwargs))
class ConfigError(ConcatError):
# docstring TODO
pass
class MasterConfig(object):
# docstring TODO
def constants(self):
'''Sets instance variables that do not change at run-time'''
self.VERSION = 'Master Server v0.1'
# A getinfo request with a challenge longer than 128 chars will be
# ignored. In practice this is far more than is necessary anyway.
self.CHALLENGE_LENGTH = 12
# This should be enough time for any decent connection but probably not
# enough for a fast typist with netcat
self.CHALLENGE_TIMEOUT = 5
# Heartbeats are usually sent every ten minutes
self.SERVER_TIMEOUT = 11 * 60
# src/client/cl_main.c -- MAX_SERVERSPERPACKET
# This limit should be hit long before the overall length limit of
# 16384 bytes
self.GSR_MAXSERVERS = 256
# Number of bytes to read from incoming packets
self.READ_SIZE = 2048
self.IGNORE_FILE = 'ignore.txt'
self.FEATURED_FILE = 'featured.txt'
self.MOTD_FILE = 'motd.txt'
def __init__(self, vlevel = LOG_PRINT):
# docstring TODO
# Set this early so that self.log can be used immediately
self.options = Values()
self.options.verbose = vlevel
def parse(self):
# docstring TODO
self.constants()
self.cmdline()
self.files()
def __getattr__(self, attr):
'''When the command line options have been parsed, this allows direct
access to them'''
# They aren't set as attributes of self directly because of the way
# optparse.OptionParser works.
return getattr(object.__getattribute__(self, 'options'), attr)
def cmdline(self):
'''Parse options from the command line. For an explanation of the
options and their usage, use:
python master.py --help
'''
# This is a m-m-m-monster function, but sometimes there are just lots
# of things to do, and that is how it is.
self.constants()
# we add our own help option for obscure reasons
parser = OptionParser(add_help_option = False)
parser.add_option('-h', '--help', action = 'store_true',
help = 'Display this help and exit')
# options other than --help are in loose alphabetical order
if has_ipv6:
parser.add_option('-4', '--ipv4', action = 'store_false',
default = True, dest = 'ipv6',
help = 'Only use IPv4')
parser.add_option('-6', '--ipv6', action = 'store_false',
default = True, dest = 'ipv4',
help = 'Only use IPv6')
# a better error message here for invalid values would be nice
parser.add_option('-d', '--db', help = 'Database backend to use, '
'<none|tdb|sqlite|auto>',
metavar = 'NAME', default = 'auto',
choices = ['none', 'tdb', 'sqlite', 'auto'])
parser.add_option('-g', '--game',
help = 'Game for which to be a master server',
metavar = 'GAME', default = 'unvanquished',
choices = ['trem', 'tremulous', 'unv', 'unvanquished'])
if has_chroot:
parser.add_option('-j', '--jail',
help = 'Path to chroot into at startup',
metavar = 'DIR')
parser.add_option('-l', '--listen-addr',
help = 'IPv4 address to listen to',
metavar = 'ADDR')
if has_ipv6:
# Can we put this in the conditional above without spoiling
# the ordering?
parser.add_option('-L', '--listen6-addr',
help = 'IPv6 address to listen to',
metavar = 'ADDR')
parser.add_option('-n', '--max-servers', type = 'int',
help = 'Maximum number of servers to track',
metavar = 'NUM')
parser.add_option('-p', '--port', type = 'int', default = -1,
help = 'Port for incoming requests',
metavar = 'NUM')
parser.add_option('-P', '--challengeport', type = 'int',
help = 'Port for outgoing challenges',
metavar = 'NUM')
parser.add_option('-q', action = 'count', default = 0,
help = 'Decrease verbose level. Multiple -q options '
'may suppress logging entirely.')
if has_setuid:
parser.add_option('-u', '--user',
help = 'User to switch to at startup')
parser.add_option('-v', action = 'count', default = 0,
help = 'Increase verbose level. Multiple -v options '
'increase the level further.')
parser.add_option('--verbose', type = 'int', default = LOG_PRINT,
help = 'Set verbose level directly. Takes a single '
'integer argument between {0} and {1}'.format(
LOG_ALWAYS, LOG_LEVELS - 1),
metavar = 'LEVEL')
parser.add_option('-V', '--version', action = 'store_true',
help = 'Show version information')
self.options, args = parser.parse_args(argv[1:])
if args:
raise ConfigError('Unexpected command line arguments')
if self.help:
stdout.write(parser.format_help())
exit(0)
# don't need this anymore
parser.destroy()
del parser
if self.game == 'trem' or self.game == 'tremulous':
self.game_name = 'Tremulous'
self.game_id = 'Tremulous'
defaultPort = 30710
else:
self.game_name = 'Unvanquished'
self.game_id = 'UNVANQUISHED'
defaultPort = 27950
self.VERSION = self.game_name + ' ' + self.VERSION
if self.version:
stdout.write('{0}\n'.format(self.VERSION))
exit(0)
self.verbose += self.v - self.q
if not LOG_ALWAYS <= self.verbose < LOG_LEVELS:
raise ConfigError('Verbose level must be between', LOG_ALWAYS,
'and', LOG_LEVELS - 1,
'(not {0})'.format(self.verbose))
self.log(LOG_VERBOSE, 'Logging:', *loglevels[:self.verbose + 1])
if not self.ipv4 and not self.ipv6:
raise ConfigError('Cannot specify both --ipv4 and --ipv6')
if self.user is not None:
try:
pwnam = getpwnam(self.user)
uid = pwnam.pw_uid
except KeyError:
try:
uid = int(self.user)
pwnam = getpwuid(uid)
except ValueError:
raise ConfigError(self.user, 'no such user', sep = ': ')
if has_setgid:
gid = pwnam.pw_gid
if self.jail is not None:
try:
chdir(self.jail)
chroot('.')
except OSError as err:
raise ConfigError('chroot {0}:'.format(self.jail),
err.strerror)
self.log(LOG_VERBOSE, 'Chrooted to', self.jail)
if self.user is not None:
# okay, probably root at this point
if has_setgroups:
try:
setgroups(())
except OSError as err:
raise ConfigError('setgroups:', err.strerror)
if has_setgid:
try:
setgid(gid)
except OSError as err:
raise ConfigError('setgid {0}:'.format(gid), err.strerror)
# setuid last, so setgroups & setgid don't fail
try:
setuid(uid)
except OSError as err:
raise ConfigError('setuid {0}:'.format(uid), err.strerror)
pwnam = None
self.log(LOG_VERBOSE, 'UID set to', getuid())
if self.port == -1:
self.port = defaultPort
if self.challengeport is None:
if self.port == 0xffff:
self.challengeport = 0xffff - 1
else:
self.challengeport = self.port + 1
self.log(LOG_VERBOSE, 'Automatically set challenge port to',
self.challengeport)
elif self.challengeport == self.port:
raise ConfigError('Request port and challenge port must not be '
'the same ({0})'.format(self.port))
def files(self):
'''Read self.FEATURED_FILE, and for each label (starting at column 0)
construct a dict of the (indented) addresses following it. Each dict
value starts off as None, to be initialised as the connections are
made.
self.featured_servers[label] is set to its corresponding dict.
A missing file is ignored but other errors - e.g. if the file is
present but can't be read - are fatal.'''
self.featured_servers = dict()
# FIXME: use ConfigError where appropriate
try:
with open(self.FEATURED_FILE) as featured:
errmsg = self.FEATURED_FILE + ':'
self.log(LOG_DEBUG, 'Opened', self.FEATURED_FILE)
label = ''
lineno = 0
for line in iter(l.rstrip() for l in featured):
lineno += 1
wheremsg = '{0}:{1}:'.format(self.FEATURED_FILE, lineno)
# ignore blank lines and comments
if not line or line.isspace() or \
line.lstrip().startswith('#'):
continue
# indented lines are server addresses
if line[0].isspace():
addr = line.lstrip()
if not label:
raise ConfigError(errmsg, 'Error: missing label')
try:
saddr = stringtosockaddr(addr)
except EnvironmentError as err:
# EnvironmentError covers socket.error and
# .gaierror without having to import them
raise ConfigError(wheremsg, "Error: couldn't "
'convert', addr,
'to address format:', err)
if saddr in self.featured_servers[label].keys():
self.log(LOG_PRINT, wheremsg, 'Warning:',
saddr, 'appears multiple times')
self.featured_servers[label][saddr] = None
# unindented lines start a new label
else:
# clean up the old label
if label:
if not self.featured_servers[label]:
raise ConfigError(wheremsg, 'Error: no '
'addresses defined for',
label)
else:
# print a message of the form
# featured.txt: 'Label': [server1, server2,...]
self.log(LOG_VERBOSE, self.FEATURED_FILE,
repr(label),
self.featured_servers[label].keys(),
sep = ': ')
label = line
for c in label:
# slashes are field seperators in
# getserversExtResponse
if c in '\\/':
raise ConfigError(wheremsg, 'Error:',
'label', repr(label), 'contains '
'invalid character:', c)
self.featured_servers[label] = dict()
if label:
# print a message of the form
# featured.txt: 'Label': [server1, server2, ...]
self.log(LOG_VERBOSE, self.FEATURED_FILE, repr(label),
self.featured_servers[label].keys(), sep = ': ')
except IOError as err:
if err.errno != ENOENT:
raise
def getmotd(self):
'''Reads the motd file and returns the contents'''
# FIXME: validate the value as an info parameter (no \\ etc.)
try:
with open(self.MOTD_FILE) as motd:
return motd.read().replace('\n','|')
except IOError as err:
if err.errno != ENOENT:
raise
def ignore(self, addr):
'''Read self.IGNORE_FILE to check if addr (an IP) is in it.
The file should consist of blank-separated addresses or address ranges
in CIDR format to ignore, with optional port or port-range.
'''
bits = inet_pton(valid_addr(addr.host), addr.host)
try:
with open(self.IGNORE_FILE) as ignore:
for line in ignore:
# comments
if line.lstrip().startswith('#'):
continue
for word in line.split():
if ':' in word:
# is there a port or port range?
host, port = word.split(':', 1)
if '-' in port:
# is this a port range?
low, high = port.split('-', 1)
try:
low, high = int(low), int(high)
except ValueError:
self.log(LOG_ERROR, "Could not parse ignored server port range in: " + word)
if addr.port < low or addr.port > high:
continue
else:
try:
port = int(port)
except ValueError:
self.log(LOG_ERROR, "Could not parse ignored server port in: " + word)
if addr.port != port:
continue
else:
host = word
try:
# is this CIDR?
net, mask = host.split('/', 1)
mask = int(mask)
except ValueError:
try:
# exact match only
if bits == inet_pton(valid_addr(host), host):
return True
except EnvironmentError as err:
# inet_pton failed
self.log(LOG_PRINT, 'ignore.txt token', word,
'could not be parsed:', err)
continue
try:
nbits = inet_pton(valid_addr(net), net)
except EnvironmentError:
continue
for byte, nbyte in zip(bits, nbits):
if not mask:
return True
elif mask >= 8:
if byte != nbyte:
return False
mask -= 8
else:
b, i = ord(byte), ord(nbyte)
m = 0xff00 >> mask
return b & m == i & m
except IOError as err:
if err.errno != ENOENT:
raise
@staticmethod
def logprefix(level):
# docstring TODO
# not sure why this is a separate method
time = strftime('%H:%M:%S')
levelname = loglevels[level]
shortlevel = levelname[:3]
return '[{time}] {levelname[0]} '.format(**locals())
def log(self, level, *args, **kwargs):
'''log(level, arg[, arg]*[, sep = ' '])
If the configuration-specified verbosity is below level, nothing
happens, otherwise a timestamp and then each str(arg) joined by the
optional keyword argument sep (default space) is printed.
IOError with errno EIO is ignored
'''
if not args:
raise TypeError('MasterConfig.log() requires at least one '
'argument')
if level > self.verbose:
return
argstr = concat(*args, **kwargs)
try:
stdout.write(self.logprefix(level) + argstr + '\n')
except IOError as err:
if err.errno == EIO:
pass
else:
raise
config = MasterConfig()
log = config.log