magic-wormhole/src/wormhole/_mailbox.py

235 lines
6.6 KiB
Python
Raw Normal View History

2018-04-21 07:30:08 +00:00
from __future__ import absolute_import, print_function, unicode_literals
from attr import attrib, attrs
2017-02-23 00:56:39 +00:00
from attr.validators import instance_of
2016-12-30 05:30:59 +00:00
from automat import MethodicalMachine
2018-04-21 07:30:08 +00:00
from zope.interface import implementer
from . import _interfaces
2016-12-30 05:30:59 +00:00
2018-04-21 07:30:08 +00:00
2017-02-23 00:56:39 +00:00
@attrs
@implementer(_interfaces.IMailbox)
class Mailbox(object):
2017-02-23 00:56:39 +00:00
_side = attrib(validator=instance_of(type(u"")))
2016-12-30 05:30:59 +00:00
m = MethodicalMachine()
2018-04-21 07:30:08 +00:00
set_trace = getattr(m, "_setTrace",
lambda self, f: None) # pragma: no cover
2016-12-30 05:30:59 +00:00
2017-02-24 02:11:07 +00:00
def __attrs_post_init__(self):
2017-02-22 20:51:53 +00:00
self._mailbox = None
self._pending_outbound = {}
self._processed = set()
def wire(self, nameplate, rendezvous_connector, ordering, terminator):
self._N = _interfaces.INameplate(nameplate)
self._RC = _interfaces.IRendezvousConnector(rendezvous_connector)
self._O = _interfaces.IOrder(ordering)
self._T = _interfaces.ITerminator(terminator)
2017-02-14 07:12:57 +00:00
# all -A states: not connected
# all -B states: yes connected
# B states serialize as A, so they deserialize as unconnected
# S0: know nothing
@m.state(initial=True)
2018-04-21 07:30:08 +00:00
def S0A(self):
pass # pragma: no cover
2016-12-30 05:30:59 +00:00
@m.state()
2018-04-21 07:30:08 +00:00
def S0B(self):
pass # pragma: no cover
2016-12-30 05:30:59 +00:00
# S1: mailbox known, not opened
2016-12-30 05:30:59 +00:00
@m.state()
2018-04-21 07:30:08 +00:00
def S1A(self):
pass # pragma: no cover
2017-02-14 07:12:57 +00:00
# S2: mailbox known, opened
# We've definitely tried to open the mailbox at least once, but it must
# be re-opened with each connection, because open() is also subscribe()
2016-12-30 05:30:59 +00:00
@m.state()
2018-04-21 07:30:08 +00:00
def S2A(self):
pass # pragma: no cover
2016-12-30 05:30:59 +00:00
@m.state()
2018-04-21 07:30:08 +00:00
def S2B(self):
pass # pragma: no cover
2016-12-30 05:30:59 +00:00
# S3: closing
2016-12-30 05:30:59 +00:00
@m.state()
2018-04-21 07:30:08 +00:00
def S3A(self):
pass # pragma: no cover
2016-12-30 05:30:59 +00:00
@m.state()
2018-04-21 07:30:08 +00:00
def S3B(self):
pass # pragma: no cover
2016-12-30 05:30:59 +00:00
# S4: closed. We no longer care whether we're connected or not
2018-04-21 07:30:08 +00:00
# @m.state()
# def S4A(self): pass
# @m.state()
# def S4B(self): pass
@m.state(terminal=True)
2018-04-21 07:30:08 +00:00
def S4(self):
pass # pragma: no cover
S4A = S4
S4B = S4
2016-12-30 05:30:59 +00:00
# from Terminator
2016-12-30 05:30:59 +00:00
@m.input()
2018-04-21 07:30:08 +00:00
def close(self, mood):
pass
# from Nameplate
@m.input()
2018-04-21 07:30:08 +00:00
def got_mailbox(self, mailbox):
pass
# from RendezvousConnector
2016-12-30 05:30:59 +00:00
@m.input()
2018-04-21 07:30:08 +00:00
def connected(self):
pass
2016-12-30 05:30:59 +00:00
@m.input()
2018-04-21 07:30:08 +00:00
def lost(self):
pass
2017-02-22 20:51:53 +00:00
def rx_message(self, side, phase, body):
assert isinstance(side, type("")), type(side)
assert isinstance(phase, type("")), type(phase)
assert isinstance(body, type(b"")), type(body)
if side == self._side:
2017-02-22 20:51:53 +00:00
self.rx_message_ours(phase, body)
else:
2017-02-26 12:13:57 +00:00
self.rx_message_theirs(side, phase, body)
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.input()
2018-04-21 07:30:08 +00:00
def rx_message_ours(self, phase, body):
pass
2016-12-30 05:30:59 +00:00
@m.input()
2018-04-21 07:30:08 +00:00
def rx_message_theirs(self, side, phase, body):
pass
2016-12-30 05:30:59 +00:00
@m.input()
2018-04-21 07:30:08 +00:00
def rx_closed(self):
pass
# from Send or Key
@m.input()
2017-02-25 02:30:00 +00:00
def add_message(self, phase, body):
pass
2016-12-30 05:30:59 +00:00
@m.output()
def record_mailbox(self, mailbox):
self._mailbox = mailbox
2018-04-21 07:30:08 +00:00
@m.output()
def RC_tx_open(self):
assert self._mailbox
self._RC.tx_open(self._mailbox)
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.output()
2017-02-22 20:51:53 +00:00
def queue(self, phase, body):
assert isinstance(phase, type("")), type(phase)
2017-02-25 02:30:00 +00:00
assert isinstance(body, type(b"")), (type(body), phase, body)
2017-02-22 20:51:53 +00:00
self._pending_outbound[phase] = body
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.output()
def record_mailbox_and_RC_tx_open_and_drain(self, mailbox):
self._mailbox = mailbox
self._RC.tx_open(mailbox)
self._drain()
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.output()
def drain(self):
self._drain()
2018-04-21 07:30:08 +00:00
def _drain(self):
2017-02-22 20:51:53 +00:00
for phase, body in self._pending_outbound.items():
self._RC.tx_add(phase, body)
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.output()
2017-02-22 20:51:53 +00:00
def RC_tx_add(self, phase, body):
assert isinstance(phase, type("")), type(phase)
assert isinstance(body, type(b"")), type(body)
self._RC.tx_add(phase, body)
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.output()
2017-02-26 12:13:57 +00:00
def N_release_and_accept(self, side, phase, body):
self._N.release()
if phase not in self._processed:
self._processed.add(phase)
self._O.got_message(side, phase, body)
2018-04-21 07:30:08 +00:00
2016-12-30 05:30:59 +00:00
@m.output()
def RC_tx_close(self):
assert self._mood
self._RC_tx_close()
2018-04-21 07:30:08 +00:00
def _RC_tx_close(self):
self._RC.tx_close(self._mailbox, self._mood)
@m.output()
2017-02-22 20:51:53 +00:00
def dequeue(self, phase, body):
2017-02-25 02:30:00 +00:00
self._pending_outbound.pop(phase, None)
2018-04-21 07:30:08 +00:00
2017-02-15 20:11:17 +00:00
@m.output()
def record_mood(self, mood):
self._mood = mood
2018-04-21 07:30:08 +00:00
@m.output()
def record_mood_and_RC_tx_close(self, mood):
self._mood = mood
self._RC_tx_close()
2018-04-21 07:30:08 +00:00
@m.output()
def ignore_mood_and_T_mailbox_done(self, mood):
self._T.mailbox_done()
2018-04-21 07:30:08 +00:00
@m.output()
def T_mailbox_done(self):
self._T.mailbox_done()
S0A.upon(connected, enter=S0B, outputs=[])
S0A.upon(got_mailbox, enter=S1A, outputs=[record_mailbox])
S0A.upon(add_message, enter=S0A, outputs=[queue])
S0A.upon(close, enter=S4A, outputs=[ignore_mood_and_T_mailbox_done])
S0B.upon(lost, enter=S0A, outputs=[])
S0B.upon(add_message, enter=S0B, outputs=[queue])
S0B.upon(close, enter=S4B, outputs=[ignore_mood_and_T_mailbox_done])
2018-04-21 07:30:08 +00:00
S0B.upon(
got_mailbox,
enter=S2B,
outputs=[record_mailbox_and_RC_tx_open_and_drain])
2016-12-30 05:30:59 +00:00
S1A.upon(connected, enter=S2B, outputs=[RC_tx_open, drain])
S1A.upon(add_message, enter=S1A, outputs=[queue])
S1A.upon(close, enter=S4A, outputs=[ignore_mood_and_T_mailbox_done])
2017-02-14 07:12:57 +00:00
S2A.upon(connected, enter=S2B, outputs=[RC_tx_open, drain])
S2A.upon(add_message, enter=S2A, outputs=[queue])
S2A.upon(close, enter=S3A, outputs=[record_mood])
S2B.upon(lost, enter=S2A, outputs=[])
S2B.upon(add_message, enter=S2B, outputs=[queue, RC_tx_add])
S2B.upon(rx_message_theirs, enter=S2B, outputs=[N_release_and_accept])
S2B.upon(rx_message_ours, enter=S2B, outputs=[dequeue])
S2B.upon(close, enter=S3B, outputs=[record_mood_and_RC_tx_close])
2016-12-30 05:30:59 +00:00
S3A.upon(connected, enter=S3B, outputs=[RC_tx_close])
S3B.upon(lost, enter=S3A, outputs=[])
S3B.upon(rx_closed, enter=S4B, outputs=[T_mailbox_done])
S3B.upon(add_message, enter=S3B, outputs=[])
S3B.upon(rx_message_theirs, enter=S3B, outputs=[])
S3B.upon(rx_message_ours, enter=S3B, outputs=[])
S3B.upon(close, enter=S3B, outputs=[])
2016-12-30 05:30:59 +00:00
S4A.upon(connected, enter=S4B, outputs=[])
S4B.upon(lost, enter=S4A, outputs=[])
S4.upon(add_message, enter=S4, outputs=[])
S4.upon(rx_message_theirs, enter=S4, outputs=[])
S4.upon(rx_message_ours, enter=S4, outputs=[])
S4.upon(close, enter=S4, outputs=[])