magic-wormhole/src/wormhole/wormhole.py

259 lines
9.3 KiB
Python
Raw Normal View History

from __future__ import print_function, absolute_import, unicode_literals
import os, sys
2017-02-23 23:57:24 +00:00
from attr import attrs, attrib
from zope.interface import implementer
from twisted.python import failure
from ._interfaces import IWormhole, IDeferredWormhole
from .util import bytes_to_hexstr
from .eventual import EventualQueue
from .observer import OneShotObserver, SequenceObserver
2017-02-23 01:02:01 +00:00
from .timing import DebugTiming
2017-02-22 20:51:53 +00:00
from .journal import ImmediateJournal
from ._boss import Boss
2017-03-04 09:55:42 +00:00
from ._key import derive_key
2017-04-03 21:23:03 +00:00
from .errors import NoKeyError, WormholeClosed
2017-03-04 09:55:42 +00:00
from .util import to_bytes
2016-05-21 01:49:20 +00:00
2017-02-23 23:57:24 +00:00
# We can provide different APIs to different apps:
# * Deferreds
# w.get_code().addCallback(print_code)
# w.send_message(data)
# w.get_message().addCallback(got_data)
2017-02-23 23:57:24 +00:00
# w.close().addCallback(closed)
# * delegate callbacks (better for journaled environments)
# w = wormhole(delegate=app)
# w.send_message(data)
2017-02-23 23:57:24 +00:00
# app.wormhole_got_code(code)
# app.wormhole_got_verifier(verifier)
# app.wormhole_got_versions(versions)
# app.wormhole_got_message(data)
2017-02-23 23:57:24 +00:00
# w.close()
# app.wormhole_closed()
#
# * potential delegate options
# wormhole(delegate=app, delegate_prefix="wormhole_",
# delegate_args=(args, kwargs))
@attrs
@implementer(IWormhole)
2017-02-23 23:57:24 +00:00
class _DelegatedWormhole(object):
_delegate = attrib()
2017-03-04 09:55:42 +00:00
def __attrs_post_init__(self):
self._key = None
2017-02-23 23:57:24 +00:00
def _set_boss(self, boss):
self._boss = boss
# from above
2017-02-24 02:23:55 +00:00
def allocate_code(self, code_length=2):
self._boss.allocate_code(code_length)
2017-04-19 02:49:50 +00:00
def input_code(self):
return self._boss.input_code()
2017-02-24 02:23:55 +00:00
def set_code(self, code):
self._boss.set_code(code)
## def serialize(self):
## s = {"serialized_wormhole_version": 1,
## "boss": self._boss.serialize(),
## }
## return s
def send_message(self, plaintext):
self._boss.send(plaintext)
2017-03-04 09:55:42 +00:00
def derive_key(self, purpose, length):
"""Derive a new key from the established wormhole channel for some
other purpose. This is a deterministic randomized function of the
session key and the 'purpose' string (unicode/py3-string). This
cannot be called until when_verifier() has fired, nor after close()
was called.
"""
if not isinstance(purpose, type("")): raise TypeError(type(purpose))
if not self._key: raise NoKeyError()
return derive_key(self._key, to_bytes(purpose), length)
2017-02-23 23:57:24 +00:00
def close(self):
self._boss.close()
2017-04-19 02:49:34 +00:00
def debug_set_trace(self, client_name, which="B N M S O K SK R RC L C T",
file=sys.stderr):
self._boss._set_trace(client_name, which, file)
2017-02-23 23:57:24 +00:00
# from below
def got_welcome(self, welcome):
self._delegate.wormhole_got_welcome(welcome)
2017-02-23 23:57:24 +00:00
def got_code(self, code):
self._delegate.wormhole_got_code(code)
2017-03-04 09:55:42 +00:00
def got_key(self, key):
self._delegate.wormhole_got_unverified_key(key)
2017-03-04 09:55:42 +00:00
self._key = key # for derive_key()
2017-02-23 23:57:24 +00:00
def got_verifier(self, verifier):
self._delegate.wormhole_got_verifier(verifier)
def got_versions(self, versions):
self._delegate.wormhole_got_versions(versions)
2017-02-24 02:23:55 +00:00
def received(self, plaintext):
self._delegate.wormhole_got_message(plaintext)
2017-02-23 23:57:24 +00:00
def closed(self, result):
self._delegate.wormhole_closed(result)
@implementer(IWormhole, IDeferredWormhole)
2017-02-23 23:57:24 +00:00
class _DeferredWormhole(object):
def __init__(self, eq):
self._welcome_observer = OneShotObserver(eq)
self._code_observer = OneShotObserver(eq)
2017-03-04 09:55:42 +00:00
self._key = None
self._key_observer = OneShotObserver(eq)
self._verifier_observer = OneShotObserver(eq)
self._version_observer = OneShotObserver(eq)
self._received_observer = SequenceObserver(eq)
self._closed = False
self._closed_observer = OneShotObserver(eq)
2017-02-23 02:21:47 +00:00
def _set_boss(self, boss):
self._boss = boss
# from above
def get_code(self):
# TODO: consider throwing error unless one of allocate/set/input_code
2017-04-07 02:44:27 +00:00
# was called first. It's legit to grab the Deferred before triggering
# the process that will cause it to fire, but forbidding that
# ordering would make it easier to cause programming errors that
# forget to trigger it entirely.
return self._code_observer.when_fired()
2017-02-23 02:21:47 +00:00
def get_welcome(self):
return self._welcome_observer.when_fired()
def get_unverified_key(self):
return self._key_observer.when_fired()
def get_verifier(self):
return self._verifier_observer.when_fired()
2017-02-23 02:21:47 +00:00
def get_versions(self):
return self._version_observer.when_fired()
2017-03-04 10:36:19 +00:00
def get_message(self):
return self._received_observer.when_next_event()
2017-02-24 02:23:55 +00:00
def allocate_code(self, code_length=2):
self._boss.allocate_code(code_length)
2017-03-19 22:09:26 +00:00
def input_code(self):
return self._boss.input_code()
2017-02-24 02:23:55 +00:00
def set_code(self, code):
self._boss.set_code(code)
# no .serialize in Deferred-mode
def send_message(self, plaintext):
self._boss.send(plaintext)
2017-03-04 09:55:42 +00:00
def derive_key(self, purpose, length):
"""Derive a new key from the established wormhole channel for some
other purpose. This is a deterministic randomized function of the
session key and the 'purpose' string (unicode/py3-string). This
cannot be called until when_verified() has fired, nor after close()
2017-03-04 09:55:42 +00:00
was called.
"""
if not isinstance(purpose, type("")): raise TypeError(type(purpose))
if not self._key: raise NoKeyError()
return derive_key(self._key, to_bytes(purpose), length)
2017-02-23 02:21:47 +00:00
def close(self):
# fails with WormholeError unless we established a connection
# (state=="happy"). Fails with WrongPasswordError (a subclass of
# WormholeError) if state=="scary".
d = self._closed_observer.when_fired() # maybe Failure
if not self._closed:
self._boss.close() # only need to close if it wasn't already
2017-02-25 02:30:00 +00:00
return d
2017-02-23 02:21:47 +00:00
def debug_set_trace(self, client_name,
which="B N M S O K SK R RC L A I C T",
file=sys.stderr):
self._boss._set_trace(client_name, which, file)
2017-02-23 02:21:47 +00:00
# from below
def got_welcome(self, welcome):
self._welcome_observer.fire_if_not_fired(welcome)
2017-02-23 02:21:47 +00:00
def got_code(self, code):
self._code_observer.fire_if_not_fired(code)
2017-03-04 09:55:42 +00:00
def got_key(self, key):
self._key = key # for derive_key()
self._key_observer.fire_if_not_fired(key)
2017-02-23 02:21:47 +00:00
def got_verifier(self, verifier):
self._verifier_observer.fire_if_not_fired(verifier)
def got_versions(self, versions):
self._version_observer.fire_if_not_fired(versions)
2017-02-23 02:21:47 +00:00
2017-02-24 02:23:55 +00:00
def received(self, plaintext):
self._received_observer.fire(plaintext)
2017-02-23 02:21:47 +00:00
def closed(self, result):
self._closed = True
#print("closed", result, type(result), file=sys.stderr)
if isinstance(result, Exception):
# everything pending gets an error, including close()
f = failure.Failure(result)
self._closed_observer.error(f)
2017-02-25 02:30:00 +00:00
else:
# everything pending except close() gets an error:
# w.get_code()/welcome/unverified_key/verifier/versions/message
f = failure.Failure(WormholeClosed(result))
# but w.close() only gets error if we're unhappy
self._closed_observer.fire_if_not_fired(result)
self._welcome_observer.error(f)
self._code_observer.error(f)
self._key_observer.error(f)
self._verifier_observer.error(f)
self._version_observer.error(f)
self._received_observer.fire(f)
2017-02-23 02:21:47 +00:00
2017-04-06 19:09:24 +00:00
def create(appid, relay_url, reactor, # use keyword args for everything else
versions={},
delegate=None, journal=None, tor=None,
timing=None,
stderr=sys.stderr,
_eventual_queue=None):
timing = timing or DebugTiming()
2017-02-23 02:21:47 +00:00
side = bytes_to_hexstr(os.urandom(5))
journal = journal or ImmediateJournal()
eq = _eventual_queue or EventualQueue(reactor)
2017-02-23 23:57:24 +00:00
if delegate:
w = _DelegatedWormhole(delegate)
else:
w = _DeferredWormhole(eq)
wormhole_versions = {} # will be used to indicate Wormhole capabilities
wormhole_versions["app_versions"] = versions # app-specific capabilities
2018-03-08 10:54:27 +00:00
#version to be sent with bind if provided
2018-03-09 08:31:59 +00:00
wormhole_version = u""
2018-03-08 10:54:27 +00:00
if type(versions) is dict and versions != {} :
wormhole_version = list(versions.values())[0] or ""
else:
wormhole_version = versions or ""
2018-03-09 08:31:59 +00:00
wormhole_version = wormhole_version.decode('unicode-escape')
2018-03-08 10:54:27 +00:00
implementation = "python"
b = Boss(w, side, relay_url, appid, wormhole_versions, wormhole_version,
reactor, journal, tor, timing, implementation)
2017-02-23 02:21:47 +00:00
w._set_boss(b)
b.start()
2016-05-21 01:49:20 +00:00
return w
## def from_serialized(serialized, reactor, delegate,
## journal=None, tor=None,
## timing=None, stderr=sys.stderr):
## assert serialized["serialized_wormhole_version"] == 1
## timing = timing or DebugTiming()
## w = _DelegatedWormhole(delegate)
## # now unpack state machines, including the SPAKE2 in Key
## b = Boss.from_serialized(w, serialized["boss"], reactor, journal, timing)
## w._set_boss(b)
## b.start() # ??
## raise NotImplemented
## # should the new Wormhole call got_code? only if it wasn't called before.