use 'mailbox_sides' instead of cols in 'mailboxes'
This commit is contained in:
parent
6c725e4a86
commit
41f229de87
|
@ -36,20 +36,24 @@ CREATE TABLE `nameplate_sides`
|
||||||
|
|
||||||
-- Clients exchange messages through a "mailbox", which has a long (randomly
|
-- Clients exchange messages through a "mailbox", which has a long (randomly
|
||||||
-- unique) identifier and a queue of messages.
|
-- unique) identifier and a queue of messages.
|
||||||
|
-- `id` is randomly-generated and unique across all apps.
|
||||||
CREATE TABLE `mailboxes`
|
CREATE TABLE `mailboxes`
|
||||||
(
|
(
|
||||||
`app_id` VARCHAR,
|
`app_id` VARCHAR,
|
||||||
`id` VARCHAR,
|
`id` VARCHAR PRIMARY KEY,
|
||||||
`side1` VARCHAR, -- side name, or NULL
|
`updated` INTEGER -- time of last activity, used for pruning
|
||||||
`side2` VARCHAR, -- side name, or NULL
|
|
||||||
`crowded` BOOLEAN, -- at some point, three or more sides were involved
|
|
||||||
`first_mood` VARCHAR,
|
|
||||||
-- timing data for the mailbox itself
|
|
||||||
`started` INTEGER, -- time when opened
|
|
||||||
`second` INTEGER -- time when second side opened
|
|
||||||
);
|
);
|
||||||
CREATE INDEX `mailboxes_idx` ON `mailboxes` (`app_id`, `id`);
|
CREATE INDEX `mailboxes_idx` ON `mailboxes` (`app_id`, `id`);
|
||||||
|
|
||||||
|
CREATE TABLE `mailbox_sides`
|
||||||
|
(
|
||||||
|
`mailbox_id` REFERENCES `mailboxes`(`id`),
|
||||||
|
`opened` BOOLEAN, -- True after open(), False after close()
|
||||||
|
`side` VARCHAR,
|
||||||
|
`added` INTEGER, -- time when this side first claimed the nameplate
|
||||||
|
`mood` VARCHAR
|
||||||
|
);
|
||||||
|
|
||||||
CREATE TABLE `messages`
|
CREATE TABLE `messages`
|
||||||
(
|
(
|
||||||
`app_id` VARCHAR,
|
`app_id` VARCHAR,
|
||||||
|
|
|
@ -16,33 +16,9 @@ EXPIRATION_CHECK_PERIOD = 1*HOUR
|
||||||
def generate_mailbox_id():
|
def generate_mailbox_id():
|
||||||
return base64.b32encode(os.urandom(8)).lower().strip(b"=").decode("ascii")
|
return base64.b32encode(os.urandom(8)).lower().strip(b"=").decode("ascii")
|
||||||
|
|
||||||
|
|
||||||
SideResult = namedtuple("SideResult", ["changed", "empty", "side1", "side2"])
|
|
||||||
Unchanged = SideResult(changed=False, empty=False, side1=None, side2=None)
|
|
||||||
class CrowdedError(Exception):
|
class CrowdedError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def add_side(row, new_side):
|
|
||||||
old_sides = [s for s in [row["side1"], row["side2"]] if s]
|
|
||||||
assert old_sides
|
|
||||||
if new_side in old_sides:
|
|
||||||
return Unchanged
|
|
||||||
if len(old_sides) == 2:
|
|
||||||
raise CrowdedError("too many sides for this thing")
|
|
||||||
return SideResult(changed=True, empty=False,
|
|
||||||
side1=old_sides[0], side2=new_side)
|
|
||||||
|
|
||||||
def remove_side(row, side):
|
|
||||||
old_sides = [s for s in [row["side1"], row["side2"]] if s]
|
|
||||||
if side not in old_sides:
|
|
||||||
return Unchanged
|
|
||||||
remaining_sides = old_sides[:]
|
|
||||||
remaining_sides.remove(side)
|
|
||||||
if remaining_sides:
|
|
||||||
return SideResult(changed=True, empty=False, side1=remaining_sides[0],
|
|
||||||
side2=None)
|
|
||||||
return SideResult(changed=True, empty=True, side1=None, side2=None)
|
|
||||||
|
|
||||||
Usage = namedtuple("Usage", ["started", "waiting_time", "total_time", "result"])
|
Usage = namedtuple("Usage", ["started", "waiting_time", "total_time", "result"])
|
||||||
TransitUsage = namedtuple("TransitUsage",
|
TransitUsage = namedtuple("TransitUsage",
|
||||||
["started", "waiting_time", "total_time",
|
["started", "waiting_time", "total_time",
|
||||||
|
@ -65,23 +41,24 @@ class Mailbox:
|
||||||
# requires caller to db.commit()
|
# requires caller to db.commit()
|
||||||
assert isinstance(side, type("")), type(side)
|
assert isinstance(side, type("")), type(side)
|
||||||
db = self._db
|
db = self._db
|
||||||
row = db.execute("SELECT * FROM `mailboxes`"
|
|
||||||
" WHERE `app_id`=? AND `id`=?",
|
already = db.execute("SELECT * FROM `mailbox_sides`"
|
||||||
(self._app_id, self._mailbox_id)).fetchone()
|
" WHERE `mailbox_id`=? AND `side`=?",
|
||||||
try:
|
(self._mailbox_id, side)).fetchone()
|
||||||
sr = add_side(row, side)
|
if not already:
|
||||||
except CrowdedError:
|
db.execute("INSERT INTO `mailbox_sides`"
|
||||||
db.execute("UPDATE `mailboxes` SET `crowded`=?"
|
" (`mailbox_id`, `opened`, `side`, `added`)"
|
||||||
" WHERE `app_id`=? AND `id`=?",
|
" VALUES(?,?,?,?)",
|
||||||
(True, self._app_id, self._mailbox_id))
|
(self._mailbox_id, True, side, when))
|
||||||
db.commit()
|
db.execute("UPDATE `mailboxes` SET `updated`=? WHERE `id`=?",
|
||||||
raise
|
(when, self._mailbox_id))
|
||||||
if sr.changed:
|
db.commit() # XXX: reconcile the need for this with the comment above
|
||||||
db.execute("UPDATE `mailboxes` SET"
|
|
||||||
" `side1`=?, `side2`=?, `second`=?"
|
rows = db.execute("SELECT * FROM `mailbox_sides`"
|
||||||
" WHERE `app_id`=? AND `id`=?",
|
" WHERE `mailbox_id`=?",
|
||||||
(sr.side1, sr.side2, when,
|
(self._mailbox_id,)).fetchall()
|
||||||
self._app_id, self._mailbox_id))
|
if len(rows) > 2:
|
||||||
|
raise CrowdedError("too many sides have opened this mailbox")
|
||||||
|
|
||||||
def get_messages(self):
|
def get_messages(self):
|
||||||
messages = []
|
messages = []
|
||||||
|
@ -118,6 +95,8 @@ class Mailbox:
|
||||||
" VALUES (?,?,?,?,?, ?,?)",
|
" VALUES (?,?,?,?,?, ?,?)",
|
||||||
(self._app_id, self._mailbox_id, sm.side,
|
(self._app_id, self._mailbox_id, sm.side,
|
||||||
sm.phase, sm.body, sm.server_rx, sm.msg_id))
|
sm.phase, sm.body, sm.server_rx, sm.msg_id))
|
||||||
|
self._db.execute("UPDATE `mailboxes` SET `updated`=? WHERE `id`=?",
|
||||||
|
(sm.server_rx, self._mailbox_id))
|
||||||
self._db.commit()
|
self._db.commit()
|
||||||
|
|
||||||
def add_message(self, sm):
|
def add_message(self, sm):
|
||||||
|
@ -133,34 +112,36 @@ class Mailbox:
|
||||||
(self._app_id, self._mailbox_id)).fetchone()
|
(self._app_id, self._mailbox_id)).fetchone()
|
||||||
if not row:
|
if not row:
|
||||||
return
|
return
|
||||||
sr = remove_side(row, side)
|
|
||||||
if sr.empty:
|
|
||||||
self._app._summarize_mailbox_and_store(self._mailbox_id, row,
|
|
||||||
mood, when, pruned=False)
|
|
||||||
self._delete()
|
|
||||||
db.commit()
|
|
||||||
elif sr.changed:
|
|
||||||
db.execute("UPDATE `mailboxes`"
|
|
||||||
" SET `side1`=?, `side2`=?, `first_mood`=?"
|
|
||||||
" WHERE `app_id`=? AND `id`=?",
|
|
||||||
(sr.side1, sr.side2, mood,
|
|
||||||
self._app_id, self._mailbox_id))
|
|
||||||
db.commit()
|
|
||||||
|
|
||||||
def _delete(self):
|
row = db.execute("SELECT * FROM `mailbox_sides`"
|
||||||
# requires caller to db.commit()
|
" WHERE `mailbox_id`=? AND `side`=?",
|
||||||
self._db.execute("DELETE FROM `mailboxes`"
|
(self._mailbox_id, side)).fetchone()
|
||||||
" WHERE `app_id`=? AND `id`=?",
|
if not row:
|
||||||
(self._app_id, self._mailbox_id))
|
return
|
||||||
self._db.execute("DELETE FROM `messages`"
|
db.execute("UPDATE `mailbox_sides` SET `opened`=?, `mood`=?"
|
||||||
" WHERE `app_id`=? AND `mailbox_id`=?",
|
" WHERE `mailbox_id`=? AND `side`=?",
|
||||||
(self._app_id, self._mailbox_id))
|
(False, mood, self._mailbox_id, side))
|
||||||
|
db.commit()
|
||||||
|
|
||||||
|
# are any sides still open?
|
||||||
|
side_rows = db.execute("SELECT * FROM `mailbox_sides`"
|
||||||
|
" WHERE `mailbox_id`=?",
|
||||||
|
(self._mailbox_id,)).fetchall()
|
||||||
|
if any([sr["opened"] for sr in side_rows]):
|
||||||
|
return
|
||||||
|
|
||||||
|
# nope. delete and summarize
|
||||||
|
db.execute("DELETE FROM `messages` WHERE `mailbox_id`=?",
|
||||||
|
(self._mailbox_id,))
|
||||||
|
db.execute("DELETE FROM `mailbox_sides` WHERE `mailbox_id`=?",
|
||||||
|
(self._mailbox_id,))
|
||||||
|
db.execute("DELETE FROM `mailboxes` WHERE `id`=?", (self._mailbox_id,))
|
||||||
|
self._app._summarize_mailbox_and_store(side_rows, when, pruned=False)
|
||||||
|
db.commit()
|
||||||
# Shut down any listeners, just in case they're still lingering
|
# Shut down any listeners, just in case they're still lingering
|
||||||
# around.
|
# around.
|
||||||
for (send_f, stop_f) in self._listeners.values():
|
for (send_f, stop_f) in self._listeners.values():
|
||||||
stop_f()
|
stop_f()
|
||||||
|
|
||||||
self._app.free_mailbox(self._mailbox_id)
|
self._app.free_mailbox(self._mailbox_id)
|
||||||
|
|
||||||
def is_active(self):
|
def is_active(self):
|
||||||
|
@ -260,6 +241,7 @@ class AppNamespace:
|
||||||
db.execute("UPDATE `nameplates` SET `updated`=? WHERE `id`=?",
|
db.execute("UPDATE `nameplates` SET `updated`=? WHERE `id`=?",
|
||||||
(when, npid))
|
(when, npid))
|
||||||
db.commit()
|
db.commit()
|
||||||
|
|
||||||
rows = db.execute("SELECT * FROM `nameplate_sides`"
|
rows = db.execute("SELECT * FROM `nameplate_sides`"
|
||||||
" WHERE `nameplates_id`=?", (npid,)).fetchall()
|
" WHERE `nameplates_id`=?", (npid,)).fetchall()
|
||||||
if len(rows) > 2:
|
if len(rows) > 2:
|
||||||
|
@ -343,11 +325,11 @@ class AppNamespace:
|
||||||
log.msg("spawning #%s for app_id %s" % (mailbox_id,
|
log.msg("spawning #%s for app_id %s" % (mailbox_id,
|
||||||
self._app_id))
|
self._app_id))
|
||||||
db.execute("INSERT INTO `mailboxes`"
|
db.execute("INSERT INTO `mailboxes`"
|
||||||
" (`app_id`, `id`, `side1`, `crowded`, `started`)"
|
" (`app_id`, `id`, `updated`)"
|
||||||
" VALUES(?,?,?,?,?)",
|
" VALUES(?,?,?)",
|
||||||
(self._app_id, mailbox_id, side, False, when))
|
(self._app_id, mailbox_id, when))
|
||||||
db.commit() # XXX
|
# we don't need a commit here, because mailbox.open() only
|
||||||
# mailbox.open() does a SELECT to find the old sides
|
# does SELECT FROM `mailbox_sides`, not from `mailboxes`
|
||||||
self._mailboxes[mailbox_id] = Mailbox(self, self._db,
|
self._mailboxes[mailbox_id] = Mailbox(self, self._db,
|
||||||
self._app_id, mailbox_id)
|
self._app_id, mailbox_id)
|
||||||
mailbox = self._mailboxes[mailbox_id]
|
mailbox = self._mailboxes[mailbox_id]
|
||||||
|
@ -365,15 +347,9 @@ class AppNamespace:
|
||||||
# log.msg("freed+killed #%s, now have %d DB mailboxes, %d live" %
|
# log.msg("freed+killed #%s, now have %d DB mailboxes, %d live" %
|
||||||
# (mailbox_id, len(self.get_claimed()), len(self._mailboxes)))
|
# (mailbox_id, len(self.get_claimed()), len(self._mailboxes)))
|
||||||
|
|
||||||
def _summarize_mailbox_and_store(self, mailbox_id, row,
|
def _summarize_mailbox_and_store(self, side_rows, delete_time, pruned):
|
||||||
second_mood, delete_time, pruned):
|
|
||||||
db = self._db
|
db = self._db
|
||||||
rows = db.execute("SELECT DISTINCT(`side`) FROM `messages`"
|
u = self._summarize_mailbox(side_rows, delete_time, pruned)
|
||||||
" WHERE `app_id`=? AND `mailbox_id`=?",
|
|
||||||
(self._app_id, mailbox_id)).fetchall()
|
|
||||||
num_sides = len(rows)
|
|
||||||
u = self._summarize_mailbox(row, num_sides, second_mood, delete_time,
|
|
||||||
pruned)
|
|
||||||
db.execute("INSERT INTO `mailbox_usage`"
|
db.execute("INSERT INTO `mailbox_usage`"
|
||||||
" (`app_id`,"
|
" (`app_id`,"
|
||||||
" `started`, `total_time`, `waiting_time`, `result`)"
|
" `started`, `total_time`, `waiting_time`, `result`)"
|
||||||
|
@ -381,16 +357,17 @@ class AppNamespace:
|
||||||
(self._app_id,
|
(self._app_id,
|
||||||
u.started, u.total_time, u.waiting_time, u.result))
|
u.started, u.total_time, u.waiting_time, u.result))
|
||||||
|
|
||||||
def _summarize_mailbox(self, row, num_sides, second_mood, delete_time,
|
def _summarize_mailbox(self, side_rows, delete_time, pruned):
|
||||||
pruned):
|
times = sorted([row["added"] for row in side_rows])
|
||||||
started = row["started"]
|
started = times[0]
|
||||||
if self._blur_usage:
|
if self._blur_usage:
|
||||||
started = self._blur_usage * (started // self._blur_usage)
|
started = self._blur_usage * (started // self._blur_usage)
|
||||||
waiting_time = None
|
waiting_time = None
|
||||||
if row["second"]:
|
if len(times) > 1:
|
||||||
waiting_time = row["second"] - row["started"]
|
waiting_time = times[1] - times[0]
|
||||||
total_time = delete_time - row["started"]
|
total_time = delete_time - times[0]
|
||||||
|
|
||||||
|
num_sides = len(times)
|
||||||
if num_sides == 0:
|
if num_sides == 0:
|
||||||
result = "quiet"
|
result = "quiet"
|
||||||
elif num_sides == 1:
|
elif num_sides == 1:
|
||||||
|
@ -398,7 +375,8 @@ class AppNamespace:
|
||||||
else:
|
else:
|
||||||
result = "happy"
|
result = "happy"
|
||||||
|
|
||||||
moods = set([row["first_mood"], second_mood])
|
# "mood" is only recorded at close()
|
||||||
|
moods = [row["mood"] for row in side_rows if row.get("mood")]
|
||||||
if "lonely" in moods:
|
if "lonely" in moods:
|
||||||
result = "lonely"
|
result = "lonely"
|
||||||
if "errory" in moods:
|
if "errory" in moods:
|
||||||
|
@ -407,7 +385,7 @@ class AppNamespace:
|
||||||
result = "scary"
|
result = "scary"
|
||||||
if pruned:
|
if pruned:
|
||||||
result = "pruney"
|
result = "pruney"
|
||||||
if row["crowded"]:
|
if num_sides > 2:
|
||||||
result = "crowded"
|
result = "crowded"
|
||||||
|
|
||||||
return Usage(started=started, waiting_time=waiting_time,
|
return Usage(started=started, waiting_time=waiting_time,
|
||||||
|
@ -425,47 +403,15 @@ class AppNamespace:
|
||||||
# for all `mailboxes`: classify as new or old
|
# for all `mailboxes`: classify as new or old
|
||||||
OLD = 0; NEW = 1
|
OLD = 0; NEW = 1
|
||||||
all_mailboxes = {}
|
all_mailboxes = {}
|
||||||
all_mailbox_rows = {}
|
for row in db.execute("SELECT * FROM `mailboxes` WHERE `app_id`=?",
|
||||||
for row in db.execute("SELECT * FROM `mailboxes`"
|
|
||||||
" WHERE `app_id`=?",
|
|
||||||
(self._app_id,)).fetchall():
|
(self._app_id,)).fetchall():
|
||||||
mailbox_id = row["id"]
|
mailbox_id = row["id"]
|
||||||
all_mailbox_rows[mailbox_id] = row
|
if row["updated"] > old:
|
||||||
if row["started"] > old:
|
|
||||||
which = NEW
|
|
||||||
elif row["second"] and row["second"] > old:
|
|
||||||
which = NEW
|
which = NEW
|
||||||
else:
|
else:
|
||||||
which = OLD
|
which = OLD
|
||||||
all_mailboxes[mailbox_id] = which
|
all_mailboxes[mailbox_id] = which
|
||||||
#log.msg(" 2: all_mailboxes", all_mailboxes, all_mailbox_rows)
|
#log.msg(" 2: all_mailboxes", all_mailboxes)
|
||||||
|
|
||||||
# for all mailbox ids used by `messages`:
|
|
||||||
# if there is no matching mailbox: delete the messages
|
|
||||||
# if there is at least one new message (select when>old limit 1):
|
|
||||||
# classify the mailbox as new
|
|
||||||
for row in db.execute("SELECT DISTINCT(`mailbox_id`)"
|
|
||||||
" FROM `messages`"
|
|
||||||
" WHERE `app_id`=?",
|
|
||||||
(self._app_id,)).fetchall():
|
|
||||||
mailbox_id = row["mailbox_id"]
|
|
||||||
if mailbox_id not in all_mailboxes:
|
|
||||||
log.msg(" deleting orphan messages", mailbox_id)
|
|
||||||
db.execute("DELETE FROM `messages`"
|
|
||||||
" WHERE `app_id`=? AND `mailbox_id`=?",
|
|
||||||
(self._app_id, mailbox_id))
|
|
||||||
modified = True
|
|
||||||
else:
|
|
||||||
new_msgs = db.execute("SELECT * FROM `messages`"
|
|
||||||
" WHERE `app_id`=? AND `mailbox_id`=?"
|
|
||||||
" AND `server_rx` > ?"
|
|
||||||
" LIMIT 1",
|
|
||||||
(self._app_id, mailbox_id, old)
|
|
||||||
).fetchall()
|
|
||||||
if new_msgs:
|
|
||||||
#log.msg(" 3-: saved by new messages", new_msgs)
|
|
||||||
all_mailboxes[mailbox_id] = NEW
|
|
||||||
#log.msg(" 4: all_mailboxes", all_mailboxes)
|
|
||||||
|
|
||||||
# for all mailbox objects with active listeners:
|
# for all mailbox objects with active listeners:
|
||||||
# classify the mailbox as new
|
# classify the mailbox as new
|
||||||
|
@ -484,8 +430,7 @@ class AppNamespace:
|
||||||
# if the nameplate is new:
|
# if the nameplate is new:
|
||||||
# classify mailbox as new
|
# classify mailbox as new
|
||||||
old_nameplate_ids = []
|
old_nameplate_ids = []
|
||||||
for row in db.execute("SELECT * FROM `nameplates`"
|
for row in db.execute("SELECT * FROM `nameplates` WHERE `app_id`=?",
|
||||||
" WHERE `app_id`=?",
|
|
||||||
(self._app_id,)).fetchall():
|
(self._app_id,)).fetchall():
|
||||||
npid = row["id"]
|
npid = row["id"]
|
||||||
if row["updated"] > old:
|
if row["updated"] > old:
|
||||||
|
@ -523,15 +468,16 @@ class AppNamespace:
|
||||||
for mailbox_id, which in all_mailboxes.items():
|
for mailbox_id, which in all_mailboxes.items():
|
||||||
if which == OLD:
|
if which == OLD:
|
||||||
log.msg(" deleting mailbox", mailbox_id)
|
log.msg(" deleting mailbox", mailbox_id)
|
||||||
self._summarize_mailbox_and_store(mailbox_id,
|
side_rows = db.execute("SELECT * FROM `mailbox_sides`"
|
||||||
all_mailbox_rows[mailbox_id],
|
" WHERE `mailbox_id`=?",
|
||||||
"pruney", now, pruned=True)
|
(mailbox_id,)).fetchall()
|
||||||
db.execute("DELETE FROM `messages`"
|
db.execute("DELETE FROM `messages` WHERE `mailbox_id`=?",
|
||||||
" WHERE `app_id`=? AND `mailbox_id`=?",
|
(mailbox_id,))
|
||||||
(self._app_id, mailbox_id))
|
db.execute("DELETE FROM `mailbox_sides` WHERE `mailbox_id`=?",
|
||||||
db.execute("DELETE FROM `mailboxes`"
|
(mailbox_id,))
|
||||||
" WHERE `app_id`=? AND `id`=?",
|
db.execute("DELETE FROM `mailboxes` WHERE `id`=?",
|
||||||
(self._app_id, mailbox_id))
|
(mailbox_id,))
|
||||||
|
self._summarize_mailbox_and_store(side_rows, now, pruned=True)
|
||||||
modified = True
|
modified = True
|
||||||
|
|
||||||
if modified:
|
if modified:
|
||||||
|
|
|
@ -12,7 +12,7 @@ from autobahn.twisted import websocket
|
||||||
from .. import __version__
|
from .. import __version__
|
||||||
from .common import ServerBase
|
from .common import ServerBase
|
||||||
from ..server import rendezvous, transit_server
|
from ..server import rendezvous, transit_server
|
||||||
from ..server.rendezvous import Usage, SidedMessage, Mailbox
|
from ..server.rendezvous import Usage, SidedMessage
|
||||||
from ..server.database import get_db
|
from ..server.database import get_db
|
||||||
|
|
||||||
class Server(ServerBase, unittest.TestCase):
|
class Server(ServerBase, unittest.TestCase):
|
||||||
|
@ -152,77 +152,69 @@ class Server(ServerBase, unittest.TestCase):
|
||||||
|
|
||||||
|
|
||||||
def _mailbox(self, app, mailbox_id):
|
def _mailbox(self, app, mailbox_id):
|
||||||
return app._db.execute("SELECT * FROM `mailboxes`"
|
mb_row = app._db.execute("SELECT * FROM `mailboxes`"
|
||||||
" WHERE `app_id`='appid' AND `id`=?",
|
" WHERE `app_id`='appid' AND `id`=?",
|
||||||
(mailbox_id,)).fetchone()
|
(mailbox_id,)).fetchone()
|
||||||
|
if not mb_row:
|
||||||
|
return None, None
|
||||||
|
side_rows = app._db.execute("SELECT * FROM `mailbox_sides`"
|
||||||
|
" WHERE `mailbox_id`=?",
|
||||||
|
(mailbox_id,)).fetchall()
|
||||||
|
return mb_row, side_rows
|
||||||
|
|
||||||
def test_mailbox(self):
|
def test_mailbox(self):
|
||||||
app = self._rendezvous.get_app("appid")
|
app = self._rendezvous.get_app("appid")
|
||||||
mailbox_id = "mid"
|
mailbox_id = "mid"
|
||||||
m1 = app.open_mailbox(mailbox_id, "side1", 0)
|
m1 = app.open_mailbox(mailbox_id, "side1", 0)
|
||||||
|
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side1")
|
self.assertEqual(len(side_rows), 1)
|
||||||
self.assertEqual(row["side2"], None)
|
self.assertEqual(side_rows[0]["side"], "side1")
|
||||||
self.assertEqual(row["crowded"], False)
|
self.assertEqual(side_rows[0]["added"], 0)
|
||||||
self.assertEqual(row["started"], 0)
|
|
||||||
self.assertEqual(row["second"], None)
|
|
||||||
|
|
||||||
# opening the same mailbox twice, by the same side, gets the same
|
# opening the same mailbox twice, by the same side, gets the same
|
||||||
# object
|
# object, and does not update the "added" timestamp
|
||||||
self.assertIdentical(m1, app.open_mailbox(mailbox_id, "side1", 1))
|
self.assertIdentical(m1, app.open_mailbox(mailbox_id, "side1", 1))
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side1")
|
self.assertEqual(len(side_rows), 1)
|
||||||
self.assertEqual(row["side2"], None)
|
self.assertEqual(side_rows[0]["side"], "side1")
|
||||||
self.assertEqual(row["crowded"], False)
|
self.assertEqual(side_rows[0]["added"], 0)
|
||||||
self.assertEqual(row["started"], 0)
|
|
||||||
self.assertEqual(row["second"], None)
|
|
||||||
|
|
||||||
# opening a second side gets the same object, and adds a new claim
|
# opening a second side gets the same object, and adds a new claim
|
||||||
self.assertIdentical(m1, app.open_mailbox(mailbox_id, "side2", 2))
|
self.assertIdentical(m1, app.open_mailbox(mailbox_id, "side2", 2))
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side1")
|
self.assertEqual(len(side_rows), 2)
|
||||||
self.assertEqual(row["side2"], "side2")
|
adds = [(row["side"], row["added"]) for row in side_rows]
|
||||||
self.assertEqual(row["crowded"], False)
|
self.assertIn(("side1", 0), adds)
|
||||||
self.assertEqual(row["started"], 0)
|
self.assertIn(("side2", 2), adds)
|
||||||
self.assertEqual(row["second"], 2)
|
|
||||||
|
|
||||||
# a third open marks it as crowded
|
# a third open marks it as crowded
|
||||||
self.assertRaises(rendezvous.CrowdedError,
|
self.assertRaises(rendezvous.CrowdedError,
|
||||||
app.open_mailbox, mailbox_id, "side3", 3)
|
app.open_mailbox, mailbox_id, "side3", 3)
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side1")
|
self.assertEqual(len(side_rows), 3)
|
||||||
self.assertEqual(row["side2"], "side2")
|
m1.close("side3", "company", 4)
|
||||||
self.assertEqual(row["crowded"], True)
|
|
||||||
self.assertEqual(row["started"], 0)
|
|
||||||
self.assertEqual(row["second"], 2)
|
|
||||||
|
|
||||||
# closing a side that never claimed the mailbox is ignored
|
# closing a side that never claimed the mailbox is ignored
|
||||||
m1.close("side4", "mood", 4)
|
m1.close("side4", "mood", 4)
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side1")
|
self.assertEqual(len(side_rows), 3)
|
||||||
self.assertEqual(row["side2"], "side2")
|
|
||||||
self.assertEqual(row["crowded"], True)
|
|
||||||
self.assertEqual(row["started"], 0)
|
|
||||||
self.assertEqual(row["second"], 2)
|
|
||||||
|
|
||||||
# closing one side leaves the second claim
|
# closing one side leaves the second claim
|
||||||
m1.close("side1", "mood", 5)
|
m1.close("side1", "mood", 5)
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side2")
|
sides = [(row["side"], row["opened"], row["mood"]) for row in side_rows]
|
||||||
self.assertEqual(row["side2"], None)
|
self.assertIn(("side1", False, "mood"), sides)
|
||||||
self.assertEqual(row["crowded"], True)
|
self.assertIn(("side2", True, None), sides)
|
||||||
self.assertEqual(row["started"], 0)
|
self.assertIn(("side3", False, "company"), sides)
|
||||||
self.assertEqual(row["second"], 2)
|
|
||||||
|
|
||||||
# closing one side multiple is ignored
|
# closing one side multiple times is ignored
|
||||||
m1.close("side1", "mood", 6)
|
m1.close("side1", "mood", 6)
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row["side1"], "side2")
|
sides = [(row["side"], row["opened"], row["mood"]) for row in side_rows]
|
||||||
self.assertEqual(row["side2"], None)
|
self.assertIn(("side1", False, "mood"), sides)
|
||||||
self.assertEqual(row["crowded"], True)
|
self.assertIn(("side2", True, None), sides)
|
||||||
self.assertEqual(row["started"], 0)
|
self.assertIn(("side3", False, "company"), sides)
|
||||||
self.assertEqual(row["second"], 2)
|
|
||||||
|
|
||||||
l1 = []; stop1 = []; stop1_f = lambda: stop1.append(True)
|
l1 = []; stop1 = []; stop1_f = lambda: stop1.append(True)
|
||||||
m1.add_listener("handle1", l1.append, stop1_f)
|
m1.add_listener("handle1", l1.append, stop1_f)
|
||||||
|
@ -231,8 +223,8 @@ class Server(ServerBase, unittest.TestCase):
|
||||||
m1.close("side2", "mood", 7)
|
m1.close("side2", "mood", 7)
|
||||||
self.assertEqual(stop1, [True])
|
self.assertEqual(stop1, [True])
|
||||||
|
|
||||||
row = self._mailbox(app, mailbox_id)
|
mb_row, side_rows = self._mailbox(app, mailbox_id)
|
||||||
self.assertEqual(row, None)
|
self.assertEqual(mb_row, None)
|
||||||
usage = app._db.execute("SELECT * FROM `mailbox_usage`").fetchone()
|
usage = app._db.execute("SELECT * FROM `mailbox_usage`").fetchone()
|
||||||
self.assertEqual(usage["app_id"], "appid")
|
self.assertEqual(usage["app_id"], "appid")
|
||||||
self.assertEqual(usage["started"], 0)
|
self.assertEqual(usage["started"], 0)
|
||||||
|
@ -384,22 +376,20 @@ class Prune(unittest.TestCase):
|
||||||
def test_lots(self):
|
def test_lots(self):
|
||||||
OLD = "old"; NEW = "new"
|
OLD = "old"; NEW = "new"
|
||||||
for nameplate in [None, OLD, NEW]:
|
for nameplate in [None, OLD, NEW]:
|
||||||
for mailbox in [None, OLD, NEW]:
|
for mailbox in [OLD, NEW]:
|
||||||
listeners = [False]
|
listeners = [False]
|
||||||
if mailbox is not None:
|
if mailbox is not None:
|
||||||
listeners = [False, True]
|
listeners = [False, True]
|
||||||
for has_listeners in listeners:
|
for has_listeners in listeners:
|
||||||
for messages in [None, OLD, NEW]:
|
self.one(nameplate, mailbox, has_listeners)
|
||||||
self.one(nameplate, mailbox, has_listeners, messages)
|
|
||||||
|
|
||||||
# def test_one(self):
|
def test_one(self):
|
||||||
# # to debug specific problems found by test_lots
|
# to debug specific problems found by test_lots
|
||||||
# self.one(None, "old", False, 'new')
|
self.one(None, "new", False)
|
||||||
|
|
||||||
def one(self, nameplate, mailbox, has_listeners, messages):
|
def one(self, nameplate, mailbox, has_listeners):
|
||||||
desc = ("nameplate=%s, mailbox=%s, has_listeners=%s,"
|
desc = ("nameplate=%s, mailbox=%s, has_listeners=%s" %
|
||||||
" messages=%s" %
|
(nameplate, mailbox, has_listeners))
|
||||||
(nameplate, mailbox, has_listeners, messages))
|
|
||||||
log.msg(desc)
|
log.msg(desc)
|
||||||
|
|
||||||
db = get_db(":memory:")
|
db = get_db(":memory:")
|
||||||
|
@ -412,46 +402,27 @@ class Prune(unittest.TestCase):
|
||||||
when = {OLD: 1, NEW: 60}
|
when = {OLD: 1, NEW: 60}
|
||||||
nameplate_survives = False
|
nameplate_survives = False
|
||||||
mailbox_survives = False
|
mailbox_survives = False
|
||||||
messages_survive = False
|
|
||||||
|
|
||||||
mbid = "mbid"
|
mbid = "mbid"
|
||||||
if nameplate is not None:
|
if nameplate is not None:
|
||||||
app.claim_nameplate("npid", "side1", when[nameplate],
|
app.claim_nameplate("npid", "side1", when[nameplate],
|
||||||
_test_mailbox_id=mbid)
|
_test_mailbox_id=mbid)
|
||||||
if mailbox is not None:
|
mb = app.open_mailbox(mbid, "side1", when[mailbox])
|
||||||
mb = app.open_mailbox(mbid, "side1", when[mailbox])
|
|
||||||
else:
|
|
||||||
# We might want a Mailbox, because that's the easiest way to add
|
|
||||||
# a "messages" row, but we can't use app.open_mailbox() because
|
|
||||||
# that modifies both the "mailboxes" table and app._mailboxes,
|
|
||||||
# and sometimes we're testing what happens when there are
|
|
||||||
# messages but not a mailbox
|
|
||||||
mb = Mailbox(app, db, APPID, mbid)
|
|
||||||
# we need app._mailboxes to know about this, because that's
|
|
||||||
# where it looks to find listeners
|
|
||||||
app._mailboxes[mbid] = mb
|
|
||||||
|
|
||||||
if messages is not None:
|
# the pruning algorithm doesn't care about the age of messages,
|
||||||
sm = SidedMessage("side1", "phase", "body", when[messages],
|
# because mailbox.updated is always updated each time we add a
|
||||||
"msgid")
|
# message
|
||||||
mb.add_message(sm)
|
sm = SidedMessage("side1", "phase", "body", when[mailbox], "msgid")
|
||||||
|
mb.add_message(sm)
|
||||||
|
|
||||||
if has_listeners:
|
if has_listeners:
|
||||||
mb.add_listener("handle", None, None)
|
mb.add_listener("handle", None, None)
|
||||||
|
|
||||||
if mailbox is None and messages is not None:
|
if (nameplate == NEW or mailbox == NEW or has_listeners):
|
||||||
# orphaned messages, even new ones, can't keep a nameplate alive
|
|
||||||
messages = None
|
|
||||||
messages_survive = False
|
|
||||||
|
|
||||||
if (nameplate == NEW or mailbox == NEW
|
|
||||||
or has_listeners or messages == NEW):
|
|
||||||
if nameplate is not None:
|
if nameplate is not None:
|
||||||
nameplate_survives = True
|
nameplate_survives = True
|
||||||
if mailbox is not None:
|
mailbox_survives = True
|
||||||
mailbox_survives = True
|
messages_survive = mailbox_survives
|
||||||
if messages is not None:
|
|
||||||
messages_survive = True
|
|
||||||
|
|
||||||
rv.prune(now=123, old=50)
|
rv.prune(now=123, old=50)
|
||||||
|
|
||||||
|
@ -931,41 +902,45 @@ class Summary(unittest.TestCase):
|
||||||
def test_mailbox(self):
|
def test_mailbox(self):
|
||||||
app = rendezvous.AppNamespace(None, None, False, None)
|
app = rendezvous.AppNamespace(None, None, False, None)
|
||||||
# starts at time 1, maybe gets second open at time 3, closes at 5
|
# starts at time 1, maybe gets second open at time 3, closes at 5
|
||||||
base_row = {"started": 1, "second": None,
|
def s(rows, pruned=False):
|
||||||
"first_mood": None, "crowded": False}
|
return app._summarize_mailbox(rows, 5, pruned)
|
||||||
def summ(num_sides, second_mood=None, pruned=False, **kwargs):
|
|
||||||
row = base_row.copy()
|
|
||||||
row.update(kwargs)
|
|
||||||
return app._summarize_mailbox(row, num_sides, second_mood, 5,
|
|
||||||
pruned)
|
|
||||||
|
|
||||||
self.assertEqual(summ(1), Usage(1, None, 4, "lonely"))
|
rows = [dict(added=1)]
|
||||||
self.assertEqual(summ(1, "lonely"), Usage(1, None, 4, "lonely"))
|
self.assertEqual(s(rows), Usage(1, None, 4, "lonely"))
|
||||||
self.assertEqual(summ(1, "errory"), Usage(1, None, 4, "errory"))
|
rows = [dict(added=1, mood="lonely")]
|
||||||
self.assertEqual(summ(1, crowded=True), Usage(1, None, 4, "crowded"))
|
self.assertEqual(s(rows), Usage(1, None, 4, "lonely"))
|
||||||
|
rows = [dict(added=1, mood="errory")]
|
||||||
|
self.assertEqual(s(rows), Usage(1, None, 4, "errory"))
|
||||||
|
rows = [dict(added=1, mood=None)]
|
||||||
|
self.assertEqual(s(rows, pruned=True), Usage(1, None, 4, "pruney"))
|
||||||
|
rows = [dict(added=1, mood="happy")]
|
||||||
|
self.assertEqual(s(rows, pruned=True), Usage(1, None, 4, "pruney"))
|
||||||
|
|
||||||
self.assertEqual(summ(2, first_mood="happy",
|
rows = [dict(added=1, mood="happy"), dict(added=3, mood="happy")]
|
||||||
second=3, second_mood="happy"),
|
self.assertEqual(s(rows), Usage(1, 2, 4, "happy"))
|
||||||
Usage(1, 2, 4, "happy"))
|
|
||||||
|
|
||||||
self.assertEqual(summ(2, first_mood="errory",
|
rows = [dict(added=1, mood="errory"), dict(added=3, mood="happy")]
|
||||||
second=3, second_mood="happy"),
|
self.assertEqual(s(rows), Usage(1, 2, 4, "errory"))
|
||||||
Usage(1, 2, 4, "errory"))
|
|
||||||
|
|
||||||
self.assertEqual(summ(2, first_mood="happy",
|
rows = [dict(added=1, mood="happy"), dict(added=3, mood="errory")]
|
||||||
second=3, second_mood="errory"),
|
self.assertEqual(s(rows), Usage(1, 2, 4, "errory"))
|
||||||
Usage(1, 2, 4, "errory"))
|
|
||||||
|
|
||||||
self.assertEqual(summ(2, first_mood="scary",
|
rows = [dict(added=1, mood="scary"), dict(added=3, mood="happy")]
|
||||||
second=3, second_mood="happy"),
|
self.assertEqual(s(rows), Usage(1, 2, 4, "scary"))
|
||||||
Usage(1, 2, 4, "scary"))
|
|
||||||
|
|
||||||
self.assertEqual(summ(2, first_mood="scary",
|
rows = [dict(added=1, mood="scary"), dict(added=3, mood="errory")]
|
||||||
second=3, second_mood="errory"),
|
self.assertEqual(s(rows), Usage(1, 2, 4, "scary"))
|
||||||
Usage(1, 2, 4, "scary"))
|
|
||||||
|
|
||||||
self.assertEqual(summ(2, first_mood="happy", second=3, pruned=True),
|
rows = [dict(added=1, mood="happy"), dict(added=3, mood=None)]
|
||||||
Usage(1, 2, 4, "pruney"))
|
self.assertEqual(s(rows, pruned=True), Usage(1, 2, 4, "pruney"))
|
||||||
|
rows = [dict(added=1, mood="happy"), dict(added=3, mood="happy")]
|
||||||
|
self.assertEqual(s(rows, pruned=True), Usage(1, 2, 4, "pruney"))
|
||||||
|
|
||||||
|
rows = [dict(added=1), dict(added=3), dict(added=4)]
|
||||||
|
self.assertEqual(s(rows), Usage(1, 2, 4, "crowded"))
|
||||||
|
|
||||||
|
rows = [dict(added=1), dict(added=3), dict(added=4)]
|
||||||
|
self.assertEqual(s(rows, pruned=True), Usage(1, 2, 4, "crowded"))
|
||||||
|
|
||||||
def test_nameplate(self):
|
def test_nameplate(self):
|
||||||
a = rendezvous.AppNamespace(None, None, False, None)
|
a = rendezvous.AppNamespace(None, None, False, None)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user