2015-03-02 08:09:17 +00:00
|
|
|
from __future__ import print_function
|
2015-10-07 00:20:12 +00:00
|
|
|
import json, time, random
|
2015-02-11 09:05:11 +00:00
|
|
|
from twisted.python import log
|
2015-10-04 22:49:06 +00:00
|
|
|
from twisted.application import service, internet
|
2015-10-07 00:20:12 +00:00
|
|
|
from twisted.web import server, resource
|
2015-02-11 09:05:11 +00:00
|
|
|
|
2015-02-12 02:13:54 +00:00
|
|
|
SECONDS = 1.0
|
2015-03-03 05:22:56 +00:00
|
|
|
MINUTE = 60*SECONDS
|
|
|
|
HOUR = 60*MINUTE
|
2015-05-05 01:13:14 +00:00
|
|
|
DAY = 24*HOUR
|
2015-02-12 02:13:54 +00:00
|
|
|
MB = 1000*1000
|
|
|
|
|
2015-05-05 01:19:40 +00:00
|
|
|
CHANNEL_EXPIRATION_TIME = 3*DAY
|
2015-10-04 22:49:06 +00:00
|
|
|
EXPIRATION_CHECK_PERIOD = 2*HOUR
|
2015-03-03 05:22:56 +00:00
|
|
|
|
2015-11-14 02:22:37 +00:00
|
|
|
ALLOCATE = u"_allocate"
|
|
|
|
DEALLOCATE = u"_deallocate"
|
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def json_response(request, data):
|
|
|
|
request.setHeader(b"content-type", b"application/json; charset=utf-8")
|
|
|
|
return (json.dumps(data)+"\n").encode("utf-8")
|
|
|
|
|
2015-03-13 06:07:47 +00:00
|
|
|
class EventsProtocol:
|
|
|
|
def __init__(self, request):
|
|
|
|
self.request = request
|
|
|
|
|
|
|
|
def sendComment(self, comment):
|
|
|
|
# this is ignored by clients, but can keep the connection open in the
|
|
|
|
# face of firewall/NAT timeouts. It also helps unit tests, since
|
|
|
|
# apparently twisted.web.client.Agent doesn't consider the connection
|
|
|
|
# to be established until it sees the first byte of the reponse body.
|
2015-09-28 23:04:54 +00:00
|
|
|
self.request.write(b": " + comment + b"\n\n")
|
2015-03-13 06:07:47 +00:00
|
|
|
|
|
|
|
def sendEvent(self, data, name=None, id=None, retry=None):
|
|
|
|
if name:
|
2015-09-28 23:04:54 +00:00
|
|
|
self.request.write(b"event: " + name.encode("utf-8") + b"\n")
|
2015-03-13 06:07:47 +00:00
|
|
|
# e.g. if name=foo, then the client web page should do:
|
|
|
|
# (new EventSource(url)).addEventListener("foo", handlerfunc)
|
|
|
|
# Note that this basically defaults to "message".
|
|
|
|
if id:
|
2015-09-28 23:04:54 +00:00
|
|
|
self.request.write(b"id: " + id.encode("utf-8") + b"\n")
|
2015-03-13 06:07:47 +00:00
|
|
|
if retry:
|
2015-09-28 23:04:54 +00:00
|
|
|
self.request.write(b"retry: " + retry + b"\n") # milliseconds
|
2015-03-13 06:07:47 +00:00
|
|
|
for line in data.splitlines():
|
2015-09-28 23:04:54 +00:00
|
|
|
self.request.write(b"data: " + line.encode("utf-8") + b"\n")
|
2015-09-27 23:55:46 +00:00
|
|
|
self.request.write(b"\n")
|
2015-03-13 06:07:47 +00:00
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
self.request.finish()
|
|
|
|
|
|
|
|
# note: no versions of IE (including the current IE11) support EventSource
|
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
# relay URLs are as follows: (MESSAGES=[{phase:,body:}..])
|
|
|
|
# GET /list?appid= -> {channelids: [INT..]}
|
|
|
|
# POST /allocate {appid:,side:} -> {channelid: INT}
|
|
|
|
# these return all messages (base64) for appid=/channelid= :
|
|
|
|
# POST /add {appid:,channelid:,side:,phase:,body:} -> {messages: MESSAGES}
|
|
|
|
# GET /get?appid=&channelid= (no-eventsource) -> {messages: MESSAGES}
|
|
|
|
# GET /get?appid=&channelid= (eventsource) -> {phase:, body:}..
|
|
|
|
# POST /deallocate {appid:,channelid:,side:} -> {status: waiting | deleted}
|
2015-10-03 19:36:14 +00:00
|
|
|
# all JSON responses include a "welcome:{..}" key
|
2015-03-22 18:45:16 +00:00
|
|
|
|
2015-11-14 02:12:47 +00:00
|
|
|
class RelayResource(resource.Resource):
|
|
|
|
def __init__(self, relay, welcome):
|
2015-02-11 09:05:11 +00:00
|
|
|
resource.Resource.__init__(self)
|
2015-10-07 00:20:12 +00:00
|
|
|
self._relay = relay
|
2015-11-14 02:12:47 +00:00
|
|
|
self._welcome = welcome
|
2015-10-03 19:36:14 +00:00
|
|
|
|
2015-11-14 02:12:47 +00:00
|
|
|
class ChannelLister(RelayResource):
|
2015-10-07 00:20:12 +00:00
|
|
|
def render_GET(self, request):
|
2015-11-11 05:02:44 +00:00
|
|
|
if b"appid" not in request.args:
|
2015-11-14 02:12:47 +00:00
|
|
|
e = NeedToUpgradeErrorResource(self._welcome)
|
2015-11-11 05:02:44 +00:00
|
|
|
return e.get_message()
|
2015-10-07 00:20:12 +00:00
|
|
|
appid = request.args[b"appid"][0].decode("utf-8")
|
|
|
|
#print("LIST", appid)
|
|
|
|
app = self._relay.get_app(appid)
|
|
|
|
allocated = app.get_allocated()
|
2015-10-03 19:36:14 +00:00
|
|
|
request.setHeader(b"content-type", b"application/json; charset=utf-8")
|
2015-11-14 02:12:47 +00:00
|
|
|
data = {"welcome": self._welcome, "channelids": sorted(allocated)}
|
2015-10-03 19:36:14 +00:00
|
|
|
return (json.dumps(data)+"\n").encode("utf-8")
|
2015-03-13 06:07:47 +00:00
|
|
|
|
2015-11-14 02:12:47 +00:00
|
|
|
class Allocator(RelayResource):
|
2015-02-11 09:05:11 +00:00
|
|
|
def render_POST(self, request):
|
2015-10-03 19:36:14 +00:00
|
|
|
content = request.content.read()
|
|
|
|
data = json.loads(content.decode("utf-8"))
|
2015-10-07 00:20:12 +00:00
|
|
|
appid = data["appid"]
|
|
|
|
side = data["side"]
|
|
|
|
if not isinstance(side, type(u"")):
|
|
|
|
raise TypeError("side must be string, not '%s'" % type(side))
|
|
|
|
#print("ALLOCATE", appid, side)
|
|
|
|
app = self._relay.get_app(appid)
|
|
|
|
channelid = app.find_available_channelid()
|
|
|
|
app.allocate_channel(channelid, side)
|
|
|
|
log.msg("allocated #%d, now have %d DB channels" %
|
|
|
|
(channelid, len(app.get_allocated())))
|
|
|
|
request.setHeader(b"content-type", b"application/json; charset=utf-8")
|
2015-11-14 02:12:47 +00:00
|
|
|
data = {"welcome": self._welcome, "channelid": channelid}
|
2015-10-07 00:20:12 +00:00
|
|
|
return (json.dumps(data)+"\n").encode("utf-8")
|
|
|
|
|
2015-11-11 05:02:44 +00:00
|
|
|
def getChild(self, path, req):
|
|
|
|
# wormhole-0.4.0 "send" started with "POST /allocate/SIDE".
|
|
|
|
# wormhole-0.5.0 changed that to "POST /allocate". We catch the old
|
|
|
|
# URL here to deliver a nicer error message (with upgrade
|
|
|
|
# instructions) than an ugly 404.
|
2015-11-14 02:12:47 +00:00
|
|
|
return NeedToUpgradeErrorResource(self._welcome)
|
2015-11-11 05:02:44 +00:00
|
|
|
|
|
|
|
class NeedToUpgradeErrorResource(resource.Resource):
|
|
|
|
def __init__(self, welcome):
|
|
|
|
resource.Resource.__init__(self)
|
|
|
|
w = welcome.copy()
|
|
|
|
w["error"] = "Sorry, you must upgrade your client to use this server."
|
|
|
|
message = {"welcome": w}
|
|
|
|
self._message = (json.dumps(message)+"\n").encode("utf-8")
|
|
|
|
def get_message(self):
|
|
|
|
return self._message
|
|
|
|
def render_POST(self, request):
|
|
|
|
return self._message
|
|
|
|
def render_GET(self, request):
|
|
|
|
return self._message
|
|
|
|
def getChild(self, path, req):
|
|
|
|
return self
|
|
|
|
|
2015-11-14 02:12:47 +00:00
|
|
|
class Adder(RelayResource):
|
2015-10-07 00:20:12 +00:00
|
|
|
def render_POST(self, request):
|
|
|
|
#content = json.load(request.content, encoding="utf-8")
|
|
|
|
content = request.content.read()
|
|
|
|
data = json.loads(content.decode("utf-8"))
|
|
|
|
appid = data["appid"]
|
|
|
|
channelid = int(data["channelid"])
|
2015-10-03 19:36:14 +00:00
|
|
|
side = data["side"]
|
|
|
|
phase = data["phase"]
|
|
|
|
if not isinstance(phase, type(u"")):
|
|
|
|
raise TypeError("phase must be string, not %s" % type(phase))
|
|
|
|
body = data["body"]
|
2015-10-07 00:20:12 +00:00
|
|
|
#print("ADD", appid, channelid, side, phase, body)
|
2015-10-03 19:36:14 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
app = self._relay.get_app(appid)
|
|
|
|
channel = app.get_channel(channelid)
|
|
|
|
response = channel.add_message(side, phase, body)
|
2015-11-14 02:12:47 +00:00
|
|
|
# response is generated with get_messages(), so it includes both
|
|
|
|
# 'welcome' and 'messages'
|
2015-10-07 00:20:12 +00:00
|
|
|
return json_response(request, response)
|
|
|
|
|
2015-11-14 02:12:47 +00:00
|
|
|
class Getter(RelayResource):
|
2015-10-07 00:20:12 +00:00
|
|
|
def render_GET(self, request):
|
|
|
|
appid = request.args[b"appid"][0].decode("utf-8")
|
|
|
|
channelid = int(request.args[b"channelid"][0])
|
|
|
|
#print("GET", appid, channelid)
|
|
|
|
app = self._relay.get_app(appid)
|
|
|
|
channel = app.get_channel(channelid)
|
|
|
|
|
|
|
|
if b"text/event-stream" not in (request.getHeader(b"accept") or b""):
|
|
|
|
response = channel.get_messages()
|
|
|
|
return json_response(request, response)
|
|
|
|
|
|
|
|
request.setHeader(b"content-type", b"text/event-stream; charset=utf-8")
|
|
|
|
ep = EventsProtocol(request)
|
2015-11-14 02:12:47 +00:00
|
|
|
ep.sendEvent(json.dumps(self._welcome), name="welcome")
|
2015-10-07 00:20:12 +00:00
|
|
|
old_events = channel.add_listener(ep.sendEvent)
|
|
|
|
request.notifyFinish().addErrback(lambda f:
|
|
|
|
channel.remove_listener(ep.sendEvent))
|
|
|
|
for old_event in old_events:
|
|
|
|
ep.sendEvent(old_event)
|
|
|
|
return server.NOT_DONE_YET
|
2015-10-03 19:36:14 +00:00
|
|
|
|
2015-11-14 02:12:47 +00:00
|
|
|
class Deallocator(RelayResource):
|
2015-10-03 19:36:14 +00:00
|
|
|
def render_POST(self, request):
|
|
|
|
content = request.content.read()
|
|
|
|
data = json.loads(content.decode("utf-8"))
|
2015-10-07 00:20:12 +00:00
|
|
|
appid = data["appid"]
|
|
|
|
channelid = int(data["channelid"])
|
2015-10-03 19:36:14 +00:00
|
|
|
side = data["side"]
|
2015-11-14 02:22:37 +00:00
|
|
|
if not isinstance(side, type(u"")):
|
|
|
|
raise TypeError("side must be string, not '%s'" % type(side))
|
|
|
|
mood = data.get("mood")
|
2015-10-07 00:20:12 +00:00
|
|
|
#print("DEALLOCATE", appid, channelid, side)
|
2015-11-14 02:22:37 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
app = self._relay.get_app(appid)
|
2015-11-14 02:22:37 +00:00
|
|
|
channel = app.get_channel(channelid)
|
|
|
|
deleted = channel.deallocate(side, mood)
|
2015-10-07 00:20:12 +00:00
|
|
|
response = {"status": "waiting"}
|
2015-10-03 19:36:14 +00:00
|
|
|
if deleted:
|
2015-10-07 00:20:12 +00:00
|
|
|
response = {"status": "deleted"}
|
|
|
|
return json_response(request, response)
|
2015-02-11 09:05:11 +00:00
|
|
|
|
2015-11-14 02:20:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Channel:
|
|
|
|
def __init__(self, app, db, welcome, appid, channelid):
|
|
|
|
self._app = app
|
|
|
|
self._db = db
|
|
|
|
self._welcome = welcome
|
2015-10-07 00:20:12 +00:00
|
|
|
self._appid = appid
|
|
|
|
self._channelid = channelid
|
|
|
|
self._listeners = set() # callbacks that take JSONable object
|
2015-05-05 01:13:14 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def get_messages(self):
|
|
|
|
messages = []
|
2015-11-14 02:20:47 +00:00
|
|
|
db = self._db
|
2015-10-07 00:20:12 +00:00
|
|
|
for row in db.execute("SELECT * FROM `messages`"
|
|
|
|
" WHERE `appid`=? AND `channelid`=?"
|
|
|
|
" ORDER BY `when` ASC",
|
|
|
|
(self._appid, self._channelid)).fetchall():
|
2015-11-14 02:22:37 +00:00
|
|
|
if row["phase"] in (u"_allocate", u"_deallocate"):
|
|
|
|
continue
|
2015-10-07 00:20:12 +00:00
|
|
|
messages.append({"phase": row["phase"], "body": row["body"]})
|
2015-11-14 02:20:47 +00:00
|
|
|
data = {"welcome": self._welcome, "messages": messages}
|
2015-10-07 00:20:12 +00:00
|
|
|
return data
|
|
|
|
|
|
|
|
def add_listener(self, listener):
|
|
|
|
self._listeners.add(listener)
|
2015-11-14 02:20:47 +00:00
|
|
|
db = self._db
|
2015-10-07 00:20:12 +00:00
|
|
|
for row in db.execute("SELECT * FROM `messages`"
|
|
|
|
" WHERE `appid`=? AND `channelid`=?"
|
|
|
|
" ORDER BY `when` ASC",
|
|
|
|
(self._appid, self._channelid)).fetchall():
|
2015-11-14 02:22:37 +00:00
|
|
|
if row["phase"] in (u"_allocate", u"_deallocate"):
|
|
|
|
continue
|
2015-10-07 00:20:12 +00:00
|
|
|
yield json.dumps({"phase": row["phase"], "body": row["body"]})
|
|
|
|
def remove_listener(self, listener):
|
|
|
|
self._listeners.discard(listener)
|
|
|
|
|
|
|
|
def broadcast_message(self, phase, body):
|
|
|
|
data = json.dumps({"phase": phase, "body": body})
|
|
|
|
for listener in self._listeners:
|
|
|
|
listener(data)
|
|
|
|
|
2015-11-14 02:22:37 +00:00
|
|
|
def _add_message(self, side, phase, body):
|
2015-11-14 02:20:47 +00:00
|
|
|
db = self._db
|
2015-10-07 00:20:12 +00:00
|
|
|
db.execute("INSERT INTO `messages`"
|
|
|
|
" (`appid`, `channelid`, `side`, `phase`, `body`, `when`)"
|
|
|
|
" VALUES (?,?,?,?, ?,?)",
|
|
|
|
(self._appid, self._channelid, side, phase,
|
|
|
|
body, time.time()))
|
|
|
|
db.commit()
|
2015-11-14 02:22:37 +00:00
|
|
|
|
|
|
|
def allocate(self, side):
|
|
|
|
self._add_message(side, ALLOCATE, None)
|
|
|
|
|
|
|
|
def add_message(self, side, phase, body):
|
|
|
|
self._add_message(side, phase, body)
|
2015-10-07 00:20:12 +00:00
|
|
|
self.broadcast_message(phase, body)
|
|
|
|
return self.get_messages()
|
|
|
|
|
2015-11-14 02:22:37 +00:00
|
|
|
def deallocate(self, side, mood):
|
|
|
|
self._add_message(side, DEALLOCATE, mood)
|
|
|
|
db = self._db
|
|
|
|
seen = set([row["side"] for row in
|
|
|
|
db.execute("SELECT `side` FROM `messages`"
|
|
|
|
" WHERE `appid`=? AND `channelid`=?",
|
|
|
|
(self._appid, self._channelid))])
|
|
|
|
freed = set([row["side"] for row in
|
|
|
|
db.execute("SELECT `side` FROM `messages`"
|
|
|
|
" WHERE `appid`=? AND `channelid`=?"
|
|
|
|
" AND `phase`=?",
|
|
|
|
(self._appid, self._channelid, DEALLOCATE))])
|
|
|
|
if seen - freed:
|
|
|
|
return False
|
|
|
|
self.delete_and_summarize()
|
|
|
|
return True
|
|
|
|
|
|
|
|
def is_idle(self):
|
|
|
|
if self._listeners:
|
|
|
|
return False
|
|
|
|
c = self._db.execute("SELECT `when` FROM `messages`"
|
|
|
|
" WHERE `appid`=? AND `channelid`=?"
|
|
|
|
" ORDER BY `when` DESC LIMIT 1",
|
|
|
|
(self._appid, self._channelid))
|
|
|
|
rows = c.fetchall()
|
|
|
|
if not rows:
|
|
|
|
return True
|
|
|
|
old = time.time() - CHANNEL_EXPIRATION_TIME
|
|
|
|
if rows[0]["when"] < old:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def delete_and_summarize(self):
|
|
|
|
# TODO: summarize usage, write into DB
|
|
|
|
db = self._db
|
|
|
|
db.execute("DELETE FROM `messages`"
|
|
|
|
" WHERE `appid`=? AND `channelid`=?",
|
|
|
|
(self._appid, self._channelid))
|
|
|
|
db.commit()
|
|
|
|
|
|
|
|
# It'd be nice to shut down any EventSource listeners here. But we
|
|
|
|
# don't hang on to the EventsProtocol, so we can't really shut it
|
|
|
|
# down here: any listeners will stick around until they shut down
|
|
|
|
# from the client side. That will keep the Channel object in memory,
|
|
|
|
# but it won't be reachable from the AppNamespace, so no further
|
|
|
|
# messages will be sent to it. Eventually, when they close the TCP
|
|
|
|
# connection, self.remove_listener() will be called, ep.sendEvent
|
|
|
|
# will be removed from self._listeners, breaking the circular
|
|
|
|
# reference, and everything will get freed.
|
|
|
|
|
|
|
|
self._app.free_channel(self._channelid)
|
|
|
|
|
|
|
|
|
2015-11-14 02:20:47 +00:00
|
|
|
class AppNamespace:
|
|
|
|
def __init__(self, db, welcome, appid):
|
|
|
|
self._db = db
|
|
|
|
self._welcome = welcome
|
2015-10-07 00:20:12 +00:00
|
|
|
self._appid = appid
|
|
|
|
self._channels = {}
|
2015-05-05 01:13:14 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def get_allocated(self):
|
2015-11-14 02:20:47 +00:00
|
|
|
db = self._db
|
2015-11-14 02:22:37 +00:00
|
|
|
c = db.execute("SELECT DISTINCT `channelid` FROM `messages`"
|
2015-10-07 00:20:12 +00:00
|
|
|
" WHERE `appid`=?", (self._appid,))
|
|
|
|
return set([row["channelid"] for row in c.fetchall()])
|
|
|
|
|
|
|
|
def find_available_channelid(self):
|
|
|
|
allocated = self.get_allocated()
|
2015-05-05 01:13:14 +00:00
|
|
|
for size in range(1,4): # stick to 1-999 for now
|
|
|
|
available = set()
|
|
|
|
for cid in range(10**(size-1), 10**size):
|
|
|
|
if cid not in allocated:
|
|
|
|
available.add(cid)
|
|
|
|
if available:
|
|
|
|
return random.choice(list(available))
|
|
|
|
# ouch, 999 currently allocated. Try random ones for a while.
|
|
|
|
for tries in range(1000):
|
|
|
|
cid = random.randrange(1000, 1000*1000)
|
|
|
|
if cid not in allocated:
|
|
|
|
return cid
|
2015-10-07 00:20:12 +00:00
|
|
|
raise ValueError("unable to find a free channel-id")
|
2015-10-04 22:49:06 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def allocate_channel(self, channelid, side):
|
2015-11-14 02:22:37 +00:00
|
|
|
channel = self.get_channel(channelid)
|
|
|
|
channel.allocate(side)
|
|
|
|
return channel
|
2015-02-11 09:05:11 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def get_channel(self, channelid):
|
|
|
|
assert isinstance(channelid, int)
|
|
|
|
if not channelid in self._channels:
|
|
|
|
log.msg("spawning #%d for appid %s" % (channelid, self._appid))
|
2015-11-14 02:20:47 +00:00
|
|
|
self._channels[channelid] = Channel(self, self._db, self._welcome,
|
2015-10-07 00:20:12 +00:00
|
|
|
self._appid, channelid)
|
|
|
|
return self._channels[channelid]
|
2015-02-11 09:05:11 +00:00
|
|
|
|
2015-11-14 02:22:37 +00:00
|
|
|
def free_channel(self, channelid):
|
|
|
|
# called from Channel.delete_and_summarize(), which deletes any
|
|
|
|
# messages
|
2015-05-05 01:13:14 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
if channelid in self._channels:
|
|
|
|
self._channels.pop(channelid)
|
2015-05-05 01:13:14 +00:00
|
|
|
log.msg("freed+killed #%d, now have %d DB channels, %d live" %
|
2015-10-07 00:20:12 +00:00
|
|
|
(channelid, len(self.get_allocated()), len(self._channels)))
|
2015-05-05 01:13:14 +00:00
|
|
|
|
|
|
|
def prune_old_channels(self):
|
2015-11-14 02:22:37 +00:00
|
|
|
log.msg(" channel prune begins")
|
|
|
|
# a channel is deleted when there are no listeners and there have
|
|
|
|
# been no messages added in CHANNEL_EXPIRATION_TIME seconds
|
|
|
|
channels = set(self.get_allocated()) # these have messages
|
|
|
|
channels.update(self._channels) # these might have listeners
|
|
|
|
for channelid in channels:
|
|
|
|
log.msg(" channel prune checking %d" % channelid)
|
|
|
|
channel = self.get_channel(channelid)
|
|
|
|
if channel.is_idle():
|
|
|
|
log.msg(" channel prune expiring %d" % channelid)
|
|
|
|
channel.delete_and_summarize() # calls self.free_channel
|
|
|
|
log.msg(" channel prune done, %r left" % (self._channels.keys(),))
|
2015-10-07 00:20:12 +00:00
|
|
|
return bool(self._channels)
|
|
|
|
|
|
|
|
class Relay(resource.Resource, service.MultiService):
|
|
|
|
def __init__(self, db, welcome):
|
|
|
|
resource.Resource.__init__(self)
|
|
|
|
service.MultiService.__init__(self)
|
2015-11-14 02:20:47 +00:00
|
|
|
self._db = db
|
|
|
|
self._welcome = welcome
|
2015-10-07 00:20:12 +00:00
|
|
|
self._apps = {}
|
|
|
|
t = internet.TimerService(EXPIRATION_CHECK_PERIOD, self.prune)
|
|
|
|
t.setServiceParent(self)
|
2015-11-14 02:12:47 +00:00
|
|
|
self.putChild(b"list", ChannelLister(self, welcome))
|
|
|
|
self.putChild(b"allocate", Allocator(self, welcome))
|
|
|
|
self.putChild(b"add", Adder(self, welcome))
|
|
|
|
self.putChild(b"get", Getter(self, welcome))
|
|
|
|
self.putChild(b"deallocate", Deallocator(self, welcome))
|
2015-10-07 00:20:12 +00:00
|
|
|
|
2015-11-11 05:02:44 +00:00
|
|
|
def getChild(self, path, req):
|
|
|
|
# 0.4.0 used "POST /CID/SIDE/post/MSGNUM"
|
|
|
|
# 0.5.0 replaced it with "POST /add (json body)"
|
|
|
|
# give a nicer error message to old clients
|
|
|
|
if (len(req.postpath) >= 2
|
|
|
|
and req.postpath[1] in (b"post", b"poll", b"deallocate")):
|
2015-11-14 02:20:47 +00:00
|
|
|
return NeedToUpgradeErrorResource(self._welcome)
|
2015-11-11 05:02:44 +00:00
|
|
|
return resource.NoResource("No such child resource.")
|
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def get_app(self, appid):
|
|
|
|
assert isinstance(appid, type(u""))
|
|
|
|
if not appid in self._apps:
|
|
|
|
log.msg("spawning appid %s" % (appid,))
|
2015-11-14 02:20:47 +00:00
|
|
|
self._apps[appid] = AppNamespace(self._db, self._welcome, appid)
|
2015-10-07 00:20:12 +00:00
|
|
|
return self._apps[appid]
|
2015-02-11 09:05:11 +00:00
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def prune(self):
|
2015-11-14 02:22:37 +00:00
|
|
|
log.msg("beginning app prune")
|
|
|
|
c = self._db.execute("SELECT DISTINCT `appid` FROM `messages`")
|
|
|
|
apps = set([row["appid"] for row in c.fetchall()]) # these have messages
|
|
|
|
apps.update(self._apps) # these might have listeners
|
|
|
|
for appid in apps:
|
|
|
|
log.msg(" app prune checking %r" % (appid,))
|
|
|
|
still_active = self.get_app(appid).prune_old_channels()
|
2015-10-07 00:20:12 +00:00
|
|
|
if not still_active:
|
2015-11-14 02:22:37 +00:00
|
|
|
log.msg("prune pops app %r" % (appid,))
|
2015-10-07 00:20:12 +00:00
|
|
|
self._apps.pop(appid)
|
2015-11-14 02:22:37 +00:00
|
|
|
log.msg("app prune ends, %d remaining apps" % len(self._apps))
|