-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathscoresd.py
142 lines (120 loc) · 3.89 KB
/
scoresd.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
try:
import MySQLdb
except ImportError:
import pymysql as MySQLdb
import scload
import time
import crawl_utils
import sys
import signal
import query
import config
import logging
from logging import debug, info, warn, error
import pagedefs
def signal_handler(signum, frame):
info("Received signal %i, terminating!", signum)
raise KeyboardInterrupt # hacky: relies on nothing else catching this
# Can run as a daemon and tail a number of logfiles and milestones and
# update the db.
def interval_work(cursor, interval, master):
master.tail_all(cursor)
scload.update_version_info(cursor)
def check_daemon_stop():
if crawl_utils.scoresd_stop_requested():
info("Exit due to scoresd stop request.")
# wait until everything actually stops to remove the stop file
return True
return False
def stop_daemon(wait):
print("Requesting daemon stop: this may take some time.")
crawl_utils.write_scoresd_stop_request()
if wait:
sys.stdout.write("Waiting...")
sys.stdout.flush()
while crawl_utils.scoresd_stop_requested():
time.sleep(5)
sys.stdout.write(".")
sys.stdout.flush()
sys.stdout.write("\n")
print("Daemon exited!")
sys.exit(0)
def tail_logfiles(logs, milestones, interval=60):
db = scload.connect_db()
scload.init_listeners(db)
cursor = db.cursor()
scload.set_active_cursor(cursor, db)
elapsed_time = 0
master = scload.create_master_reader()
scload.bootstrap_known_raceclasses(cursor)
scload.init_game_restrictions(cursor)
daemon_loop = True
if scload.OPT.run_bans:
scload.run_bans(cursor)
pagedefs.incremental_build(cursor)
daemon_loop = False # a one-off command, don't really start the daemon
try:
while daemon_loop:
try:
interval_work(cursor, interval, master)
pagedefs.incremental_build(cursor)
if not interval:
break
except IOError as e:
error("IOError: %s" % e)
info("Finished batch.");
if check_daemon_stop():
break
if not scload.OPT.force_loop and scload.OPT.run_once:
break
if interval > 60:
info("Sleeping for %d seconds" % interval)
total_to_sleep = interval
slept = 0
while (total_to_sleep > 60):
time.sleep(60)
elapsed_time += 60
total_to_sleep = total_to_sleep - 60
if check_daemon_stop():
total_to_sleep = 0
break
if total_to_sleep > 0:
time.sleep(total_to_sleep)
elapsed_time += total_to_sleep
pagedefs.tick_dirty()
if check_daemon_stop():
break
except KeyboardInterrupt: # signal or ctrl-c in non-daemon mode
warn("Rollback triggered by interrupt signal")
cursor.db.rollback()
finally:
if not scload.OPT.load_only:
info("Flushing player pages and shutting down db connection")
try:
pagedefs.flush_pages(cursor) # flush any dirty player pages
except Exception as e:
error("Failed to flush pages: " + str(e))
scload.set_active_cursor(None)
cursor.close()
db.close()
crawl_utils.clear_scoresd_stop_request()
info("Daemon exit")
if __name__ == '__main__':
daemon = not scload.OPT.no_daemonize
signal.signal(signal.SIGTERM, signal_handler)
signal.signal(signal.SIGHUP, signal_handler) # TODO: restart on SIGHUP?
# n.b. SIGKILL may result in dirty pages not being flushed
logformat = config.LOGFORMAT
if daemon:
logging.basicConfig(level=logging.INFO,
filename = config.LOGFILE,
format = logformat)
else:
logging.basicConfig(level=logging.INFO, format = logformat)
if scload.OPT.stop_daemon:
stop_daemon(scload.OPT.stop_daemon_wait) # NORETURN
crawl_utils.clear_scoresd_stop_request() # just in case
if daemon:
crawl_utils.daemonize()
interval = config.CONFIG.get("update-interval", 60)
tail_logfiles( config.SOURCES.logfiles(), config.SOURCES.milestones(), interval )