magic-wormhole/src/wormhole/_send.py

56 lines
1.7 KiB
Python
Raw Normal View History

from zope.interface import implementer
from automat import MethodicalMachine
from . import _interfaces
from .util import hexstr_to_bytes
@implementer(_interfaces.ISend)
class Send(object):
m = MethodicalMachine()
def __init__(self, side, timing):
self._side = side
self._timing = timing
def wire(self, mailbox):
self._M = _interfaces.IMailbox(mailbox)
@m.state(initial=True)
def S0_no_key(self): pass
@m.state(terminal=True)
def S1_verified_key(self): pass
def got_pake(self, payload):
if "pake_v1" in payload:
self.got_pake_good(hexstr_to_bytes(payload["pake_v1"]))
else:
self.got_pake_bad()
@m.input()
def got_verified_key(self, key): pass
@m.input()
def send(self, phase, payload): pass
@m.output()
def queue(self, phase, payload):
self._queue.append((phase, payload))
@m.output()
def record_key(self, key):
self._key = key
@m.output()
def drain(self, key):
del key
for (phase, payload) in self._queue:
self._encrypt_and_send(phase, payload)
self._queue[:] = []
@m.output()
def deliver(self, phase, payload):
self._encrypt_and_send(phase, payload)
def _encrypt_and_send(self, phase, payload):
data_key = self._derive_phase_key(self._side, phase)
encrypted = self._encrypt_data(data_key, plaintext)
self._M.add_message(phase, encrypted)
S0_no_key.upon(send, enter=S0_no_key, outputs=[queue])
S0_no_key.upon(got_verified_key, enter=S1_verified_key,
outputs=[record_key, drain])
S1_verified_key.upon(send, enter=S1_verified_key, outputs=[deliver])