magic-wormhole/src/wormhole/_boss.py

238 lines
8.6 KiB
Python
Raw Normal View History

2017-02-24 02:11:07 +00:00
from __future__ import print_function, absolute_import, unicode_literals
2017-02-25 02:30:00 +00:00
import re
import six
from zope.interface import implementer
2017-02-23 23:57:24 +00:00
from attr import attrs, attrib
from attr.validators import provides, instance_of
2017-02-25 02:30:00 +00:00
from twisted.python import log
from automat import MethodicalMachine
from . import _interfaces
from ._mailbox import Mailbox
from ._send import Send
from ._order import Order
from ._key import Key
from ._receive import Receive
from ._rendezvous import RendezvousConnector
from ._nameplate import NameplateListing
from ._code import Code
2017-02-25 02:30:00 +00:00
from .errors import WrongPasswordError
2017-02-22 20:51:53 +00:00
from .util import bytes_to_dict
2017-02-25 02:30:00 +00:00
class WormholeError(Exception):
pass
2017-02-23 02:21:47 +00:00
@attrs
@implementer(_interfaces.IBoss)
2017-02-23 23:57:24 +00:00
class Boss(object):
_W = attrib()
2017-02-23 02:21:47 +00:00
_side = attrib(validator=instance_of(type(u"")))
_url = attrib(validator=instance_of(type(u"")))
_appid = attrib(validator=instance_of(type(u"")))
_reactor = attrib()
_journal = attrib(validator=provides(_interfaces.IJournal))
_timing = attrib(validator=provides(_interfaces.ITiming))
2017-02-15 20:11:17 +00:00
m = MethodicalMachine()
def __attrs_post_init__(self):
2017-02-23 02:21:47 +00:00
self._M = Mailbox(self._side)
self._S = Send(self._side, self._timing)
self._O = Order(self._side, self._timing)
2017-02-25 02:30:00 +00:00
self._K = Key(self._appid, self._side, self._timing)
2017-02-23 02:21:47 +00:00
self._R = Receive(self._side, self._timing)
self._RC = RendezvousConnector(self._url, self._appid, self._side,
self._reactor, self._journal,
self._timing)
self._NL = NameplateListing()
2017-02-23 02:21:47 +00:00
self._C = Code(self._timing)
self._M.wire(self, self._RC, self._O)
self._S.wire(self._M)
self._O.wire(self._K, self._R)
self._K.wire(self, self._M, self._R)
self._R.wire(self, self._K, self._S)
2017-02-22 20:51:53 +00:00
self._RC.wire(self, self._M, self._C, self._NL)
self._NL.wire(self._RC, self._C)
self._C.wire(self, self._RC, self._NL)
2017-02-15 20:11:17 +00:00
self._next_tx_phase = 0
self._next_rx_phase = 0
self._rx_phases = {} # phase -> plaintext
2017-02-25 02:30:00 +00:00
self._result = "empty"
2017-02-15 20:11:17 +00:00
# these methods are called from outside
def start(self):
2017-02-22 20:51:53 +00:00
self._RC.start()
2017-02-15 20:11:17 +00:00
# and these are the state-machine transition functions, which don't take
# args
@m.state(initial=True)
def S0_empty(self): pass
2017-02-15 20:11:17 +00:00
@m.state()
def S1_lonely(self): pass
2017-02-15 20:11:17 +00:00
@m.state()
def S2_happy(self): pass
2017-02-15 20:11:17 +00:00
@m.state()
def S3_closing(self): pass
2017-02-15 20:11:17 +00:00
@m.state(terminal=True)
def S4_closed(self): pass
2017-02-15 20:11:17 +00:00
2017-02-23 00:56:39 +00:00
# from the Wormhole
# input/allocate/set_code are regular methods, not state-transition
# inputs. We expect them to be called just after initialization, while
# we're in the S0_empty state. You must call exactly one of them, and the
# call must happen while we're in S0_empty, which makes them good
# candiates for being a proper @m.input, but set_code() will immediately
# (reentrantly) cause self.got_code() to be fired, which is messy. These
# are all passthroughs to the Code machine, so one alternative would be
# to have Wormhole call Code.{input,allocate,set_code} instead, but that
# would require the Wormhole to be aware of Code (whereas right now
# Wormhole only knows about this Boss instance, and everything else is
# hidden away).
2017-02-24 02:23:55 +00:00
def input_code(self, stdio):
self._C.input_code(stdio)
def allocate_code(self, code_length):
self._C.allocate_code(code_length)
2017-02-23 00:56:39 +00:00
def set_code(self, code):
self._C.set_code(code)
2017-02-15 20:11:17 +00:00
@m.input()
def send(self, plaintext): pass
2017-02-15 20:11:17 +00:00
@m.input()
def close(self): pass
2017-02-15 20:11:17 +00:00
2017-02-24 02:11:07 +00:00
# from RendezvousConnector
@m.input()
def rx_welcome(self, welcome): pass
2017-02-23 00:56:39 +00:00
# from Code (provoked by input/allocate/set_code)
2017-02-15 20:11:17 +00:00
@m.input()
2017-02-23 00:56:39 +00:00
def got_code(self, code): pass
2017-02-15 20:11:17 +00:00
# Key sends (got_verifier, scared)
# Receive sends (got_message, happy, scared)
2017-02-15 20:11:17 +00:00
@m.input()
def happy(self): pass
2017-02-15 20:11:17 +00:00
@m.input()
def scared(self): pass
2017-02-25 02:30:00 +00:00
@m.input()
def rx_error(self, err, orig): pass
def got_message(self, phase, plaintext):
2017-02-22 20:51:53 +00:00
assert isinstance(phase, type("")), type(phase)
assert isinstance(plaintext, type(b"")), type(plaintext)
if phase == "version":
self.got_version(plaintext)
2017-02-25 02:30:00 +00:00
elif re.search(r'^\d+$', phase):
self.got_phase(int(phase), plaintext)
else:
2017-02-25 02:30:00 +00:00
# Ignore unrecognized phases, for forwards-compatibility. Use
# log.err so tests will catch surprises.
log.err("received unknown phase '%s'" % phase)
2017-02-15 20:11:17 +00:00
@m.input()
2017-02-23 00:56:39 +00:00
def got_version(self, plaintext): pass
2017-02-15 20:11:17 +00:00
@m.input()
def got_phase(self, phase, plaintext): pass
2017-02-15 20:11:17 +00:00
@m.input()
def got_verifier(self, verifier): pass
# Mailbox sends closed
2017-02-15 20:11:17 +00:00
@m.input()
def closed(self): pass
2017-02-15 20:11:17 +00:00
2017-02-24 02:11:07 +00:00
@m.output()
def process_welcome(self, welcome):
pass # TODO: ignored for now
@m.output()
2017-02-23 00:56:39 +00:00
def do_got_code(self, code):
nameplate = code.split("-")[0]
self._M.set_nameplate(nameplate)
2017-02-23 00:56:39 +00:00
self._K.got_code(code)
2017-02-23 02:21:47 +00:00
self._W.got_code(code)
2017-02-15 20:11:17 +00:00
@m.output()
2017-02-22 20:51:53 +00:00
def process_version(self, plaintext):
self._their_versions = bytes_to_dict(plaintext)
# ignored for now
2017-02-15 20:11:17 +00:00
@m.output()
def S_send(self, plaintext):
2017-02-25 02:30:00 +00:00
assert isinstance(plaintext, type(b"")), type(plaintext)
phase = self._next_tx_phase
self._next_tx_phase += 1
2017-02-25 02:30:00 +00:00
self._S.send("%d" % phase, plaintext)
2017-02-15 20:11:17 +00:00
@m.output()
2017-02-25 02:30:00 +00:00
def close_error(self, err, orig):
self._result = WormholeError(err)
self._M.close("errory")
@m.output()
def close_scared(self):
2017-02-25 02:30:00 +00:00
self._result = WrongPasswordError()
self._M.close("scary")
2017-02-15 20:11:17 +00:00
@m.output()
def close_lonely(self):
2017-02-25 02:30:00 +00:00
self._result = WormholeError("lonely")
self._M.close("lonely")
@m.output()
def close_happy(self):
2017-02-25 02:30:00 +00:00
self._result = "happy"
self._M.close("happy")
2017-02-15 20:11:17 +00:00
@m.output()
2017-02-23 02:21:47 +00:00
def W_got_verifier(self, verifier):
self._W.got_verifier(verifier)
@m.output()
2017-02-23 02:21:47 +00:00
def W_received(self, phase, plaintext):
2017-02-25 02:30:00 +00:00
assert isinstance(phase, six.integer_types), type(phase)
# we call Wormhole.received() in strict phase order, with no gaps
self._rx_phases[phase] = plaintext
while self._next_rx_phase in self._rx_phases:
2017-02-24 02:23:55 +00:00
self._W.received(self._rx_phases.pop(self._next_rx_phase))
self._next_rx_phase += 1
2017-02-15 20:11:17 +00:00
@m.output()
2017-02-23 02:21:47 +00:00
def W_closed(self):
2017-02-25 02:30:00 +00:00
self._W.closed(self._result)
S0_empty.upon(close, enter=S3_closing, outputs=[close_lonely])
S0_empty.upon(send, enter=S0_empty, outputs=[S_send])
2017-02-24 02:11:07 +00:00
S0_empty.upon(rx_welcome, enter=S0_empty, outputs=[process_welcome])
2017-02-23 00:56:39 +00:00
S0_empty.upon(got_code, enter=S1_lonely, outputs=[do_got_code])
2017-02-25 02:30:00 +00:00
S0_empty.upon(rx_error, enter=S3_closing, outputs=[close_error])
2017-02-24 02:11:07 +00:00
S1_lonely.upon(rx_welcome, enter=S1_lonely, outputs=[process_welcome])
S1_lonely.upon(happy, enter=S2_happy, outputs=[])
S1_lonely.upon(scared, enter=S3_closing, outputs=[close_scared])
S1_lonely.upon(close, enter=S3_closing, outputs=[close_lonely])
S1_lonely.upon(send, enter=S1_lonely, outputs=[S_send])
2017-02-23 02:21:47 +00:00
S1_lonely.upon(got_verifier, enter=S1_lonely, outputs=[W_got_verifier])
2017-02-25 02:30:00 +00:00
S1_lonely.upon(rx_error, enter=S3_closing, outputs=[close_error])
2017-02-24 02:11:07 +00:00
S2_happy.upon(rx_welcome, enter=S2_happy, outputs=[process_welcome])
2017-02-23 02:21:47 +00:00
S2_happy.upon(got_phase, enter=S2_happy, outputs=[W_received])
S2_happy.upon(got_version, enter=S2_happy, outputs=[process_version])
S2_happy.upon(scared, enter=S3_closing, outputs=[close_scared])
S2_happy.upon(close, enter=S3_closing, outputs=[close_happy])
S2_happy.upon(send, enter=S2_happy, outputs=[S_send])
2017-02-25 02:30:00 +00:00
S2_happy.upon(rx_error, enter=S3_closing, outputs=[close_error])
2017-02-24 02:11:07 +00:00
S3_closing.upon(rx_welcome, enter=S3_closing, outputs=[])
2017-02-25 02:30:00 +00:00
S3_closing.upon(rx_error, enter=S3_closing, outputs=[])
S3_closing.upon(got_phase, enter=S3_closing, outputs=[])
S3_closing.upon(got_version, enter=S3_closing, outputs=[])
S3_closing.upon(happy, enter=S3_closing, outputs=[])
S3_closing.upon(scared, enter=S3_closing, outputs=[])
S3_closing.upon(close, enter=S3_closing, outputs=[])
S3_closing.upon(send, enter=S3_closing, outputs=[])
2017-02-23 02:21:47 +00:00
S3_closing.upon(closed, enter=S4_closed, outputs=[W_closed])
2017-02-24 02:11:07 +00:00
S4_closed.upon(rx_welcome, enter=S4_closed, outputs=[])
S4_closed.upon(got_phase, enter=S4_closed, outputs=[])
S4_closed.upon(got_version, enter=S4_closed, outputs=[])
S4_closed.upon(happy, enter=S4_closed, outputs=[])
S4_closed.upon(scared, enter=S4_closed, outputs=[])
S4_closed.upon(close, enter=S4_closed, outputs=[])
S4_closed.upon(send, enter=S4_closed, outputs=[])