2016-06-04 06:07:50 +00:00
|
|
|
from __future__ import print_function, absolute_import, unicode_literals
|
2016-05-29 01:19:45 +00:00
|
|
|
import os, sys, re
|
2016-05-21 01:49:20 +00:00
|
|
|
from six.moves.urllib_parse import urlparse
|
|
|
|
from twisted.internet import defer, endpoints, error
|
|
|
|
from twisted.internet.threads import deferToThread, blockingCallFromThread
|
|
|
|
from twisted.internet.defer import inlineCallbacks, returnValue
|
2016-05-26 01:05:02 +00:00
|
|
|
from twisted.python import log, failure
|
2016-05-21 01:49:20 +00:00
|
|
|
from autobahn.twisted import websocket
|
|
|
|
from nacl.secret import SecretBox
|
|
|
|
from nacl.exceptions import CryptoError
|
|
|
|
from nacl import utils
|
|
|
|
from spake2 import SPAKE2_Symmetric
|
2016-05-24 20:47:15 +00:00
|
|
|
from hashlib import sha256
|
2016-05-22 18:31:00 +00:00
|
|
|
from . import __version__
|
|
|
|
from . import codes
|
|
|
|
#from .errors import ServerError, Timeout
|
2016-06-22 08:04:05 +00:00
|
|
|
from .errors import (WrongPasswordError, InternalError, WelcomeError,
|
2016-06-02 21:12:54 +00:00
|
|
|
WormholeClosedError, KeyFormatError)
|
2016-05-23 01:40:44 +00:00
|
|
|
from .timing import DebugTiming
|
2016-05-29 01:19:45 +00:00
|
|
|
from .util import (to_bytes, bytes_to_hexstr, hexstr_to_bytes,
|
|
|
|
dict_to_bytes, bytes_to_dict)
|
2016-05-21 01:49:20 +00:00
|
|
|
from hkdf import Hkdf
|
|
|
|
|
|
|
|
def HKDF(skm, outlen, salt=None, CTXinfo=b""):
|
|
|
|
return Hkdf(salt, skm).expand(CTXinfo, outlen)
|
|
|
|
|
|
|
|
CONFMSG_NONCE_LENGTH = 128//8
|
|
|
|
CONFMSG_MAC_LENGTH = 256//8
|
|
|
|
def make_confmsg(confkey, nonce):
|
|
|
|
return nonce+HKDF(confkey, CONFMSG_MAC_LENGTH, nonce)
|
|
|
|
|
|
|
|
|
|
|
|
# We send the following messages through the relay server to the far side (by
|
|
|
|
# sending "add" commands to the server, and getting "message" responses):
|
|
|
|
#
|
|
|
|
# phase=setup:
|
|
|
|
# * unauthenticated version strings (but why?)
|
|
|
|
# * early warmup for connection hints ("I can do tor, spin up HS")
|
|
|
|
# * wordlist l10n identifier
|
|
|
|
# phase=pake: just the SPAKE2 'start' message (binary)
|
2016-05-26 02:13:37 +00:00
|
|
|
# phase=version: version data, key verification (HKDF(key, nonce)+nonce)
|
2016-05-21 01:49:20 +00:00
|
|
|
# phase=1,2,3,..: application messages
|
|
|
|
|
|
|
|
class WSClient(websocket.WebSocketClientProtocol):
|
|
|
|
def onOpen(self):
|
2016-12-15 08:04:17 +00:00
|
|
|
#self.wormhole_open = True
|
|
|
|
self.connection_machine.onOpen()
|
|
|
|
#self.factory.d.callback(self)
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
def onMessage(self, payload, isBinary):
|
|
|
|
assert not isBinary
|
|
|
|
self.wormhole._ws_dispatch_response(payload)
|
|
|
|
|
|
|
|
def onClose(self, wasClean, code, reason):
|
2016-12-15 08:04:17 +00:00
|
|
|
self.connection_machine.onClose()
|
|
|
|
#if self.wormhole_open:
|
|
|
|
# self.wormhole._ws_closed(wasClean, code, reason)
|
|
|
|
#else:
|
|
|
|
# # we closed before establishing a connection (onConnect) or
|
|
|
|
# # finishing WebSocket negotiation (onOpen): errback
|
|
|
|
# self.factory.d.errback(error.ConnectError(reason))
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
class WSFactory(websocket.WebSocketClientFactory):
|
|
|
|
protocol = WSClient
|
|
|
|
def buildProtocol(self, addr):
|
|
|
|
proto = websocket.WebSocketClientFactory.buildProtocol(self, addr)
|
2016-12-15 08:04:17 +00:00
|
|
|
proto.connection_machine = self.connection_machine
|
|
|
|
#proto.wormhole_open = False
|
2016-05-21 01:49:20 +00:00
|
|
|
return proto
|
|
|
|
|
|
|
|
|
|
|
|
class _GetCode:
|
2016-05-23 01:40:44 +00:00
|
|
|
def __init__(self, code_length, send_command, timing):
|
2016-05-21 01:49:20 +00:00
|
|
|
self._code_length = code_length
|
|
|
|
self._send_command = send_command
|
2016-05-23 01:40:44 +00:00
|
|
|
self._timing = timing
|
2016-05-21 01:49:20 +00:00
|
|
|
self._allocated_d = defer.Deferred()
|
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def go(self):
|
|
|
|
with self._timing.add("allocate"):
|
2016-06-04 19:47:51 +00:00
|
|
|
self._send_command("allocate")
|
2016-05-21 01:49:20 +00:00
|
|
|
nameplate_id = yield self._allocated_d
|
|
|
|
code = codes.make_code(nameplate_id, self._code_length)
|
2016-06-04 19:47:51 +00:00
|
|
|
assert isinstance(code, type("")), type(code)
|
2016-05-21 01:49:20 +00:00
|
|
|
returnValue(code)
|
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _response_handle_allocated(self, msg):
|
2016-05-21 01:49:20 +00:00
|
|
|
nid = msg["nameplate"]
|
2016-06-04 19:47:51 +00:00
|
|
|
assert isinstance(nid, type("")), type(nid)
|
2016-05-21 01:49:20 +00:00
|
|
|
self._allocated_d.callback(nid)
|
|
|
|
|
|
|
|
class _InputCode:
|
2017-01-06 16:25:32 +00:00
|
|
|
def __init__(self, reactor, prompt, code_length, send_command, timing,
|
|
|
|
stderr):
|
2016-05-21 01:49:20 +00:00
|
|
|
self._reactor = reactor
|
|
|
|
self._prompt = prompt
|
|
|
|
self._code_length = code_length
|
|
|
|
self._send_command = send_command
|
2016-05-23 01:40:44 +00:00
|
|
|
self._timing = timing
|
2017-01-06 16:25:32 +00:00
|
|
|
self._stderr = stderr
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def _list(self):
|
|
|
|
self._lister_d = defer.Deferred()
|
2016-06-04 19:47:51 +00:00
|
|
|
self._send_command("list")
|
2016-05-21 01:49:20 +00:00
|
|
|
nameplates = yield self._lister_d
|
|
|
|
self._lister_d = None
|
|
|
|
returnValue(nameplates)
|
|
|
|
|
|
|
|
def _list_blocking(self):
|
|
|
|
return blockingCallFromThread(self._reactor, self._list)
|
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def go(self):
|
|
|
|
# fetch the list of nameplates ahead of time, to give us a chance to
|
|
|
|
# discover the welcome message (and warn the user about an obsolete
|
|
|
|
# client)
|
|
|
|
#
|
|
|
|
# TODO: send the request early, show the prompt right away, hide the
|
|
|
|
# latency in the user's indecision and slow typing. If we're lucky
|
|
|
|
# the answer will come back before they hit TAB.
|
|
|
|
|
|
|
|
initial_nameplate_ids = yield self._list()
|
|
|
|
with self._timing.add("input code", waiting="user"):
|
|
|
|
t = self._reactor.addSystemEventTrigger("before", "shutdown",
|
|
|
|
self._warn_readline)
|
2017-01-06 17:05:34 +00:00
|
|
|
res = yield deferToThread(codes.input_code_with_completion,
|
|
|
|
self._prompt,
|
|
|
|
initial_nameplate_ids,
|
|
|
|
self._list_blocking,
|
|
|
|
self._code_length)
|
|
|
|
(code, used_completion) = res
|
2016-05-21 01:49:20 +00:00
|
|
|
self._reactor.removeSystemEventTrigger(t)
|
2017-01-06 17:05:34 +00:00
|
|
|
if not used_completion:
|
|
|
|
self._remind_about_tab()
|
2016-05-21 01:49:20 +00:00
|
|
|
returnValue(code)
|
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _response_handle_nameplates(self, msg):
|
2016-05-21 01:49:20 +00:00
|
|
|
nameplates = msg["nameplates"]
|
|
|
|
assert isinstance(nameplates, list), type(nameplates)
|
2016-05-22 18:31:00 +00:00
|
|
|
nids = []
|
|
|
|
for n in nameplates:
|
|
|
|
assert isinstance(n, dict), type(n)
|
2016-06-04 19:47:51 +00:00
|
|
|
nameplate_id = n["id"]
|
|
|
|
assert isinstance(nameplate_id, type("")), type(nameplate_id)
|
2016-05-22 18:31:00 +00:00
|
|
|
nids.append(nameplate_id)
|
|
|
|
self._lister_d.callback(nids)
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
def _warn_readline(self):
|
|
|
|
# When our process receives a SIGINT, Twisted's SIGINT handler will
|
|
|
|
# stop the reactor and wait for all threads to terminate before the
|
|
|
|
# process exits. However, if we were waiting for
|
|
|
|
# input_code_with_completion() when SIGINT happened, the readline
|
|
|
|
# thread will be blocked waiting for something on stdin. Trick the
|
|
|
|
# user into satisfying the blocking read so we can exit.
|
|
|
|
print("\nCommand interrupted: please press Return to quit",
|
|
|
|
file=sys.stderr)
|
|
|
|
|
|
|
|
# Other potential approaches to this problem:
|
|
|
|
# * hard-terminate our process with os._exit(1), but make sure the
|
|
|
|
# tty gets reset to a normal mode ("cooked"?) first, so that the
|
|
|
|
# next shell command the user types is echoed correctly
|
|
|
|
# * track down the thread (t.p.threadable.getThreadID from inside the
|
|
|
|
# thread), get a cffi binding to pthread_kill, deliver SIGINT to it
|
|
|
|
# * allocate a pty pair (pty.openpty), replace sys.stdin with the
|
|
|
|
# slave, build a pty bridge that copies bytes (and other PTY
|
|
|
|
# things) from the real stdin to the master, then close the slave
|
|
|
|
# at shutdown, so readline sees EOF
|
|
|
|
# * write tab-completion and basic editing (TTY raw mode,
|
|
|
|
# backspace-is-erase) without readline, probably with curses or
|
|
|
|
# twisted.conch.insults
|
|
|
|
# * write a separate program to get codes (maybe just "wormhole
|
|
|
|
# --internal-get-code"), run it as a subprocess, let it inherit
|
|
|
|
# stdin/stdout, send it SIGINT when we receive SIGINT ourselves. It
|
|
|
|
# needs an RPC mechanism (over some extra file descriptors) to ask
|
|
|
|
# us to fetch the current nameplate_id list.
|
|
|
|
#
|
|
|
|
# Note that hard-terminating our process with os.kill(os.getpid(),
|
|
|
|
# signal.SIGKILL), or SIGTERM, doesn't seem to work: the thread
|
|
|
|
# doesn't see the signal, and we must still wait for stdin to make
|
|
|
|
# readline finish.
|
|
|
|
|
2017-01-06 17:05:34 +00:00
|
|
|
def _remind_about_tab(self):
|
|
|
|
print(" (note: you can use <Tab> to complete words)", file=self._stderr)
|
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
class _WelcomeHandler:
|
2016-05-23 01:40:44 +00:00
|
|
|
def __init__(self, url, current_version, signal_error):
|
2016-05-22 18:31:00 +00:00
|
|
|
self._ws_url = url
|
|
|
|
self._version_warning_displayed = False
|
|
|
|
self._current_version = current_version
|
2016-05-23 01:40:44 +00:00
|
|
|
self._signal_error = signal_error
|
2016-05-22 18:31:00 +00:00
|
|
|
|
|
|
|
def handle_welcome(self, welcome):
|
2016-05-29 02:19:22 +00:00
|
|
|
if "motd" in welcome:
|
2016-05-22 18:31:00 +00:00
|
|
|
motd_lines = welcome["motd"].splitlines()
|
|
|
|
motd_formatted = "\n ".join(motd_lines)
|
|
|
|
print("Server (at %s) says:\n %s" %
|
|
|
|
(self._ws_url, motd_formatted), file=sys.stderr)
|
|
|
|
|
|
|
|
# Only warn if we're running a release version (e.g. 0.0.6, not
|
|
|
|
# 0.0.6-DISTANCE-gHASH). Only warn once.
|
2016-05-29 02:11:27 +00:00
|
|
|
if ("current_cli_version" in welcome
|
2016-05-22 18:31:00 +00:00
|
|
|
and "-" not in self._current_version
|
|
|
|
and not self._version_warning_displayed
|
2016-05-29 02:11:27 +00:00
|
|
|
and welcome["current_cli_version"] != self._current_version):
|
2016-05-22 18:31:00 +00:00
|
|
|
print("Warning: errors may occur unless both sides are running the same version", file=sys.stderr)
|
|
|
|
print("Server claims %s is current, but ours is %s"
|
2016-05-29 02:11:27 +00:00
|
|
|
% (welcome["current_cli_version"], self._current_version),
|
2016-05-22 18:31:00 +00:00
|
|
|
file=sys.stderr)
|
|
|
|
self._version_warning_displayed = True
|
|
|
|
|
|
|
|
if "error" in welcome:
|
2016-05-26 22:37:24 +00:00
|
|
|
return self._signal_error(WelcomeError(welcome["error"]),
|
2016-06-04 19:47:51 +00:00
|
|
|
"unwelcome")
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
# states for nameplates, mailboxes, and the websocket connection
|
|
|
|
(CLOSED, OPENING, OPEN, CLOSING) = ("closed", "opening", "open", "closing")
|
|
|
|
|
2016-12-15 08:04:17 +00:00
|
|
|
from automat import MethodicalMachine
|
|
|
|
# pip install (path to automat checkout)[visualize]
|
|
|
|
# automat-visualize wormhole.wormhole
|
|
|
|
|
|
|
|
class _ConnectionMachine(object):
|
|
|
|
m = MethodicalMachine()
|
|
|
|
|
|
|
|
def __init__(self, ws_url):
|
|
|
|
self._f = f = WSFactory(ws_url)
|
|
|
|
f.setProtocolOptions(autoPingInterval=60, autoPingTimeout=600)
|
|
|
|
f.connection_machine = self # calls onOpen and onClose
|
|
|
|
p = urlparse(ws_url)
|
|
|
|
self._ep = self._make_endpoint(p.hostname, p.port or 80)
|
|
|
|
self._connector = None
|
|
|
|
|
|
|
|
@m.state(initial=True)
|
|
|
|
def initial(self): pass
|
|
|
|
@m.state()
|
|
|
|
def first_time_connecting(self): pass
|
|
|
|
@m.state()
|
|
|
|
def negotiating(self): pass
|
|
|
|
@m.state()
|
|
|
|
def open(self): pass
|
|
|
|
@m.state()
|
|
|
|
def waiting(self): pass
|
|
|
|
@m.state()
|
|
|
|
def connecting(self): pass
|
|
|
|
@m.state()
|
|
|
|
def disconnecting(self): pass
|
|
|
|
@m.state()
|
|
|
|
def disconnecting2(self): pass
|
|
|
|
@m.state(terminal=True)
|
|
|
|
def failed(self): pass
|
|
|
|
@m.state(terminal=True)
|
|
|
|
def closed(self): pass
|
|
|
|
|
|
|
|
|
|
|
|
@m.input()
|
|
|
|
def start(self): pass
|
|
|
|
@m.input()
|
|
|
|
def d_callback(self, p): pass
|
|
|
|
@m.input()
|
|
|
|
def d_errback(self, f): pass
|
|
|
|
@m.input()
|
|
|
|
def d_cancel(self): pass
|
|
|
|
@m.input()
|
|
|
|
def onOpen(self, ws): pass
|
|
|
|
@m.input()
|
|
|
|
def onClose(self): pass
|
|
|
|
@m.input()
|
|
|
|
def expire(self): pass
|
|
|
|
@m.input()
|
|
|
|
def close(self): pass
|
|
|
|
|
|
|
|
@m.output()
|
|
|
|
def ep_connect(self):
|
|
|
|
"ep.connect()"
|
|
|
|
self._d = self._ep.connect(self._f)
|
|
|
|
self._d.addBoth(self.d_callback, self.d_errback)
|
|
|
|
@m.output()
|
|
|
|
def handle_connection(self, ws):
|
|
|
|
pass
|
|
|
|
@m.output()
|
|
|
|
def start_timer(self):
|
|
|
|
pass
|
|
|
|
@m.output()
|
|
|
|
def cancel_timer(self):
|
|
|
|
pass
|
|
|
|
@m.output()
|
|
|
|
def dropConnection(self):
|
|
|
|
pass
|
|
|
|
@m.output()
|
2016-12-16 23:37:34 +00:00
|
|
|
def notify_fail(self, f):
|
2016-12-15 08:04:17 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
initial.upon(start, enter=first_time_connecting, outputs=[ep_connect])
|
|
|
|
first_time_connecting.upon(d_callback, enter=negotiating, outputs=[])
|
|
|
|
first_time_connecting.upon(d_errback, enter=failed, outputs=[notify_fail])
|
|
|
|
first_time_connecting.upon(close, enter=disconnecting2, outputs=[d_cancel])
|
|
|
|
disconnecting2.upon(d_errback, enter=closed, outputs=[])
|
|
|
|
|
|
|
|
negotiating.upon(onOpen, enter=open, outputs=[handle_connection])
|
|
|
|
negotiating.upon(close, enter=disconnecting, outputs=[dropConnection])
|
|
|
|
negotiating.upon(onClose, enter=failed, outputs=[notify_fail])
|
|
|
|
|
|
|
|
open.upon(onClose, enter=waiting, outputs=[start_timer])
|
|
|
|
open.upon(close, enter=disconnecting, outputs=[dropConnection])
|
|
|
|
connecting.upon(d_callback, enter=negotiating, outputs=[])
|
|
|
|
connecting.upon(d_errback, enter=waiting, outputs=[start_timer])
|
|
|
|
connecting.upon(close, enter=disconnecting2, outputs=[d_cancel])
|
|
|
|
|
|
|
|
waiting.upon(expire, enter=connecting, outputs=[ep_connect])
|
|
|
|
waiting.upon(close, enter=closed, outputs=[cancel_timer])
|
|
|
|
disconnecting.upon(onClose, enter=closed, outputs=[])
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
class _Wormhole:
|
2016-05-26 22:49:45 +00:00
|
|
|
DEBUG = False
|
|
|
|
|
2017-01-06 16:25:32 +00:00
|
|
|
def __init__(self, appid, relay_url, reactor, tor_manager, timing, stderr):
|
2016-05-22 18:31:00 +00:00
|
|
|
self._appid = appid
|
|
|
|
self._ws_url = relay_url
|
|
|
|
self._reactor = reactor
|
|
|
|
self._tor_manager = tor_manager
|
|
|
|
self._timing = timing
|
2017-01-06 16:25:32 +00:00
|
|
|
self._stderr = stderr
|
2016-05-22 18:31:00 +00:00
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
self._welcomer = _WelcomeHandler(self._ws_url, __version__,
|
|
|
|
self._signal_error)
|
2016-05-29 01:19:45 +00:00
|
|
|
self._side = bytes_to_hexstr(os.urandom(5))
|
2016-05-24 05:53:00 +00:00
|
|
|
self._connection_state = CLOSED
|
2016-05-23 01:40:44 +00:00
|
|
|
self._connection_waiters = []
|
2016-06-04 05:55:52 +00:00
|
|
|
self._ws_t = None
|
2016-05-23 01:40:44 +00:00
|
|
|
self._started_get_code = False
|
2016-05-23 07:14:39 +00:00
|
|
|
self._get_code = None
|
2016-05-24 07:00:04 +00:00
|
|
|
self._started_input_code = False
|
2016-05-26 22:38:19 +00:00
|
|
|
self._input_code_waiter = None
|
2016-05-23 01:40:44 +00:00
|
|
|
self._code = None
|
2016-05-22 18:31:00 +00:00
|
|
|
self._nameplate_id = None
|
2016-05-24 05:53:00 +00:00
|
|
|
self._nameplate_state = CLOSED
|
2016-05-22 18:31:00 +00:00
|
|
|
self._mailbox_id = None
|
2016-05-24 05:53:00 +00:00
|
|
|
self._mailbox_state = CLOSED
|
2016-05-23 01:40:44 +00:00
|
|
|
self._flag_need_nameplate = True
|
2016-05-21 01:49:20 +00:00
|
|
|
self._flag_need_to_see_mailbox_used = True
|
|
|
|
self._flag_need_to_build_msg1 = True
|
|
|
|
self._flag_need_to_send_PAKE = True
|
2016-12-16 09:33:17 +00:00
|
|
|
self._establish_key_called = False
|
2016-06-05 06:21:44 +00:00
|
|
|
self._key_waiter = None
|
2016-05-23 01:40:44 +00:00
|
|
|
self._key = None
|
2016-05-26 01:05:02 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
self._version_message = None
|
|
|
|
self._version_checked = False
|
2016-05-26 01:05:02 +00:00
|
|
|
self._get_verifier_called = False
|
|
|
|
self._verifier = None # bytes
|
|
|
|
self._verify_result = None # bytes or a Failure
|
|
|
|
self._verifier_waiter = None
|
|
|
|
|
2016-05-26 01:27:37 +00:00
|
|
|
self._my_versions = {} # sent
|
|
|
|
self._their_versions = {} # received
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
self._close_called = False # the close() API has been called
|
|
|
|
self._closing = False # we've started shutdown
|
2016-05-23 01:45:50 +00:00
|
|
|
self._disconnect_waiter = defer.Deferred()
|
2016-05-23 07:14:39 +00:00
|
|
|
self._error = None
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-21 01:49:20 +00:00
|
|
|
self._next_send_phase = 0
|
2016-05-23 01:40:44 +00:00
|
|
|
# send() queues plaintext here, waiting for a connection and the key
|
|
|
|
self._plaintext_to_send = [] # (phase, plaintext)
|
|
|
|
self._sent_phases = set() # to detect double-send
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
self._next_receive_phase = 0
|
2016-05-22 18:31:00 +00:00
|
|
|
self._receive_waiters = {} # phase -> Deferred
|
2016-05-23 01:40:44 +00:00
|
|
|
self._received_messages = {} # phase -> plaintext
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
# API METHODS for applications to call
|
2016-05-23 07:14:39 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
# You must use at least one of these entry points, to establish the
|
|
|
|
# wormhole code. Other APIs will stall or be queued until we have one.
|
|
|
|
|
|
|
|
# entry point 1: generate a new code. returns a Deferred
|
|
|
|
def get_code(self, code_length=2): # XX rename to allocate_code()? create_?
|
|
|
|
return self._API_get_code(code_length)
|
|
|
|
|
|
|
|
# entry point 2: interactively type in a code, with completion. returns
|
|
|
|
# Deferred
|
|
|
|
def input_code(self, prompt="Enter wormhole code: ", code_length=2):
|
|
|
|
return self._API_input_code(prompt, code_length)
|
|
|
|
|
|
|
|
# entry point 3: paste in a fully-formed code. No return value.
|
|
|
|
def set_code(self, code):
|
|
|
|
self._API_set_code(code)
|
|
|
|
|
|
|
|
# todo: restore-saved-state entry points
|
|
|
|
|
2016-06-05 06:21:44 +00:00
|
|
|
def establish_key(self):
|
|
|
|
"""
|
|
|
|
returns a Deferred that fires when we've established the shared key.
|
|
|
|
When successful, the Deferred fires with a simple `True`, otherwise
|
|
|
|
it fails.
|
|
|
|
"""
|
|
|
|
return self._API_establish_key()
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def verify(self):
|
|
|
|
"""Returns a Deferred that fires when we've heard back from the other
|
|
|
|
side, and have confirmed that they used the right wormhole code. When
|
|
|
|
successful, the Deferred fires with a "verifier" (a bytestring) which
|
|
|
|
can be compared out-of-band before making additional API calls. If
|
|
|
|
they used the wrong wormhole code, the Deferred errbacks with
|
|
|
|
WrongPasswordError.
|
|
|
|
"""
|
|
|
|
return self._API_verify()
|
|
|
|
|
|
|
|
def send(self, outbound_data):
|
|
|
|
return self._API_send(outbound_data)
|
|
|
|
|
|
|
|
def get(self):
|
|
|
|
return self._API_get()
|
|
|
|
|
|
|
|
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 verify() or get() has fired.
|
|
|
|
"""
|
|
|
|
return self._API_derive_key(purpose, length)
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
def close(self, res=None):
|
|
|
|
"""Collapse the wormhole, freeing up server resources and flushing
|
|
|
|
all pending messages. Returns a Deferred that fires when everything
|
|
|
|
is done. It fires with any argument close() was given, to enable use
|
|
|
|
as a d.addBoth() handler:
|
|
|
|
|
|
|
|
w = wormhole(...)
|
|
|
|
d = w.get()
|
|
|
|
..
|
|
|
|
d.addBoth(w.close)
|
|
|
|
return d
|
|
|
|
|
|
|
|
Another reasonable approach is to use inlineCallbacks:
|
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def pair(self, code):
|
|
|
|
w = wormhole(...)
|
|
|
|
try:
|
|
|
|
them = yield w.get()
|
|
|
|
finally:
|
|
|
|
yield w.close()
|
|
|
|
"""
|
|
|
|
return self._API_close(res)
|
2016-05-24 05:53:00 +00:00
|
|
|
|
|
|
|
# INTERNAL METHODS beyond here
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
def _start(self):
|
|
|
|
d = self._connect() # causes stuff to happen
|
|
|
|
d.addErrback(log.err)
|
|
|
|
return d # fires when connection is established, if you care
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
|
|
|
|
|
2016-05-21 01:49:20 +00:00
|
|
|
def _make_endpoint(self, hostname, port):
|
|
|
|
if self._tor_manager:
|
|
|
|
return self._tor_manager.get_endpoint_for(hostname, port)
|
|
|
|
# note: HostnameEndpoints have a default 30s timeout
|
|
|
|
return endpoints.HostnameEndpoint(self._reactor, hostname, port)
|
|
|
|
|
|
|
|
def _connect(self):
|
|
|
|
# TODO: if we lose the connection, make a new one, re-establish the
|
|
|
|
# state
|
|
|
|
assert self._side
|
2016-05-24 05:53:00 +00:00
|
|
|
self._connection_state = OPENING
|
2016-06-04 05:55:52 +00:00
|
|
|
self._ws_t = self._timing.add("open websocket")
|
2016-05-21 01:49:20 +00:00
|
|
|
p = urlparse(self._ws_url)
|
|
|
|
f = WSFactory(self._ws_url)
|
2016-07-04 04:51:56 +00:00
|
|
|
f.setProtocolOptions(autoPingInterval=60, autoPingTimeout=600)
|
2016-05-21 01:49:20 +00:00
|
|
|
f.wormhole = self
|
|
|
|
f.d = defer.Deferred()
|
|
|
|
# TODO: if hostname="localhost", I get three factories starting
|
|
|
|
# and stopping (maybe 127.0.0.1, ::1, and something else?), and
|
|
|
|
# an error in the factory is masked.
|
|
|
|
ep = self._make_endpoint(p.hostname, p.port or 80)
|
|
|
|
# .connect errbacks if the TCP connection fails
|
|
|
|
d = ep.connect(f)
|
|
|
|
d.addCallback(self._event_connected)
|
|
|
|
# f.d is errbacked if WebSocket negotiation fails, and the WebSocket
|
|
|
|
# drops any data sent before onOpen() fires, so we must wait for it
|
2016-05-22 18:31:00 +00:00
|
|
|
d.addCallback(lambda _: f.d)
|
2016-05-21 01:49:20 +00:00
|
|
|
d.addCallback(self._event_ws_opened)
|
|
|
|
return d
|
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _event_connected(self, ws):
|
2016-05-21 01:49:20 +00:00
|
|
|
self._ws = ws
|
2016-06-04 06:30:31 +00:00
|
|
|
if self._ws_t:
|
|
|
|
self._ws_t.finish()
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
def _event_ws_opened(self, _):
|
2016-05-24 05:53:00 +00:00
|
|
|
self._connection_state = OPEN
|
|
|
|
if self._closing:
|
|
|
|
return self._maybe_finished_closing()
|
2016-06-04 19:47:51 +00:00
|
|
|
self._ws_send_command("bind", appid=self._appid, side=self._side)
|
2016-05-24 05:53:00 +00:00
|
|
|
self._maybe_claim_nameplate()
|
2016-05-22 18:31:00 +00:00
|
|
|
self._maybe_send_pake()
|
2016-05-23 01:40:44 +00:00
|
|
|
waiters, self._connection_waiters = self._connection_waiters, []
|
|
|
|
for d in waiters:
|
|
|
|
d.callback(None)
|
|
|
|
|
|
|
|
def _when_connected(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
if self._connection_state == OPEN:
|
2016-05-23 01:40:44 +00:00
|
|
|
return defer.succeed(None)
|
|
|
|
d = defer.Deferred()
|
|
|
|
self._connection_waiters.append(d)
|
|
|
|
return d
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _ws_send_command(self, mtype, **kwargs):
|
|
|
|
# msgid is used by misc/dump-timing.py to correlate our sends with
|
|
|
|
# their receives, and vice versa. They are also correlated with the
|
|
|
|
# ACKs we get back from the server (which we otherwise ignore). There
|
|
|
|
# are so few messages, 16 bits is enough to be mostly-unique.
|
2016-05-24 05:53:00 +00:00
|
|
|
if self.DEBUG: print("SEND", mtype)
|
2016-05-29 01:19:45 +00:00
|
|
|
kwargs["id"] = bytes_to_hexstr(os.urandom(2))
|
2016-05-22 18:31:00 +00:00
|
|
|
kwargs["type"] = mtype
|
2016-05-29 01:19:45 +00:00
|
|
|
payload = dict_to_bytes(kwargs)
|
2016-05-22 18:31:00 +00:00
|
|
|
self._timing.add("ws_send", _side=self._side, **kwargs)
|
|
|
|
self._ws.sendMessage(payload, False)
|
|
|
|
|
|
|
|
def _ws_dispatch_response(self, payload):
|
2016-05-29 01:19:45 +00:00
|
|
|
msg = bytes_to_dict(payload)
|
2016-05-23 01:40:44 +00:00
|
|
|
if self.DEBUG and msg["type"]!="ack": print("DIS", msg["type"], msg)
|
2016-05-22 18:31:00 +00:00
|
|
|
self._timing.add("ws_receive", _side=self._side, message=msg)
|
|
|
|
mtype = msg["type"]
|
|
|
|
meth = getattr(self, "_response_handle_"+mtype, None)
|
|
|
|
if not meth:
|
|
|
|
# make tests fail, but real application will ignore it
|
|
|
|
log.err(ValueError("Unknown inbound message type %r" % (msg,)))
|
|
|
|
return
|
|
|
|
return meth(msg)
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _response_handle_ack(self, msg):
|
|
|
|
pass
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _response_handle_welcome(self, msg):
|
|
|
|
self._welcomer.handle_welcome(msg["welcome"])
|
2016-05-21 01:49:20 +00:00
|
|
|
|
|
|
|
# entry point 1: generate a new code
|
|
|
|
@inlineCallbacks
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_get_code(self, code_length):
|
2016-06-22 08:04:05 +00:00
|
|
|
if self._code is not None: raise InternalError
|
|
|
|
if self._started_get_code: raise InternalError
|
2016-05-21 01:49:20 +00:00
|
|
|
self._started_get_code = True
|
|
|
|
with self._timing.add("API get_code"):
|
2016-05-23 01:40:44 +00:00
|
|
|
yield self._when_connected()
|
|
|
|
gc = _GetCode(code_length, self._ws_send_command, self._timing)
|
2016-05-23 07:14:39 +00:00
|
|
|
self._get_code = gc
|
2016-05-22 18:31:00 +00:00
|
|
|
self._response_handle_allocated = gc._response_handle_allocated
|
2016-05-23 07:14:39 +00:00
|
|
|
# TODO: signal_error
|
2016-05-21 01:49:20 +00:00
|
|
|
code = yield gc.go()
|
2016-05-23 07:14:39 +00:00
|
|
|
self._get_code = None
|
2016-05-24 05:53:00 +00:00
|
|
|
self._nameplate_state = OPEN
|
2016-05-21 01:49:20 +00:00
|
|
|
self._event_learned_code(code)
|
|
|
|
returnValue(code)
|
|
|
|
|
|
|
|
# entry point 2: interactively type in a code, with completion
|
|
|
|
@inlineCallbacks
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_input_code(self, prompt, code_length):
|
2016-06-22 08:04:05 +00:00
|
|
|
if self._code is not None: raise InternalError
|
|
|
|
if self._started_input_code: raise InternalError
|
2016-05-21 01:49:20 +00:00
|
|
|
self._started_input_code = True
|
|
|
|
with self._timing.add("API input_code"):
|
2016-05-23 01:40:44 +00:00
|
|
|
yield self._when_connected()
|
2016-05-24 07:00:04 +00:00
|
|
|
ic = _InputCode(self._reactor, prompt, code_length,
|
2017-01-06 16:25:32 +00:00
|
|
|
self._ws_send_command, self._timing, self._stderr)
|
2016-05-22 18:31:00 +00:00
|
|
|
self._response_handle_nameplates = ic._response_handle_nameplates
|
2016-05-26 22:38:19 +00:00
|
|
|
# we reveal the Deferred we're waiting on, so _signal_error can
|
|
|
|
# wake us up if something goes wrong (like a welcome error)
|
|
|
|
self._input_code_waiter = ic.go()
|
|
|
|
code = yield self._input_code_waiter
|
|
|
|
self._input_code_waiter = None
|
2016-05-21 01:49:20 +00:00
|
|
|
self._event_learned_code(code)
|
|
|
|
returnValue(None)
|
|
|
|
|
|
|
|
# entry point 3: paste in a fully-formed code
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_set_code(self, code):
|
2016-05-21 01:49:20 +00:00
|
|
|
self._timing.add("API set_code")
|
2016-06-03 22:17:47 +00:00
|
|
|
if not isinstance(code, type(u"")):
|
|
|
|
raise TypeError("Unexpected code type '{}'".format(type(code)))
|
|
|
|
if self._code is not None:
|
2016-06-22 08:04:05 +00:00
|
|
|
raise InternalError
|
2016-05-21 01:49:20 +00:00
|
|
|
self._event_learned_code(code)
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
# TODO: entry point 4: restore pre-contact saved state (we haven't heard
|
|
|
|
# from the peer yet, so we still need the nameplate)
|
|
|
|
|
|
|
|
# TODO: entry point 5: restore post-contact saved state (so we don't need
|
|
|
|
# or use the nameplate, only the mailbox)
|
|
|
|
def _restore_post_contact_state(self, state):
|
|
|
|
# ...
|
|
|
|
self._flag_need_nameplate = False
|
|
|
|
#self._mailbox_id = X(state)
|
|
|
|
self._event_learned_mailbox()
|
|
|
|
|
2016-05-21 01:49:20 +00:00
|
|
|
def _event_learned_code(self, code):
|
|
|
|
self._timing.add("code established")
|
2016-06-02 21:12:54 +00:00
|
|
|
# bail out early if the password contains spaces...
|
|
|
|
# this should raise a useful error
|
|
|
|
if ' ' in code:
|
2016-06-02 21:21:29 +00:00
|
|
|
raise KeyFormatError("code (%s) contains spaces." % code)
|
2016-05-21 01:49:20 +00:00
|
|
|
self._code = code
|
|
|
|
mo = re.search(r'^(\d+)-', code)
|
|
|
|
if not mo:
|
|
|
|
raise ValueError("code (%s) must start with NN-" % code)
|
|
|
|
nid = mo.group(1)
|
2016-06-04 19:47:51 +00:00
|
|
|
assert isinstance(nid, type("")), type(nid)
|
2016-05-21 01:49:20 +00:00
|
|
|
self._nameplate_id = nid
|
|
|
|
# fire more events
|
|
|
|
self._maybe_build_msg1()
|
|
|
|
self._event_learned_nameplate()
|
|
|
|
|
|
|
|
def _maybe_build_msg1(self):
|
|
|
|
if not (self._code and self._flag_need_to_build_msg1):
|
|
|
|
return
|
|
|
|
with self._timing.add("pake1", waiting="crypto"):
|
|
|
|
self._sp = SPAKE2_Symmetric(to_bytes(self._code),
|
|
|
|
idSymmetric=to_bytes(self._appid))
|
|
|
|
self._msg1 = self._sp.start()
|
|
|
|
self._flag_need_to_build_msg1 = False
|
|
|
|
self._event_built_msg1()
|
|
|
|
|
|
|
|
def _event_built_msg1(self):
|
|
|
|
self._maybe_send_pake()
|
|
|
|
|
|
|
|
# every _maybe_X starts with a set of conditions
|
|
|
|
# for each such condition Y, every _event_Y must call _maybe_X
|
|
|
|
|
|
|
|
def _event_learned_nameplate(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
self._maybe_claim_nameplate()
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _maybe_claim_nameplate(self):
|
|
|
|
if not (self._nameplate_id and self._connection_state == OPEN):
|
2016-05-21 01:49:20 +00:00
|
|
|
return
|
2016-06-04 19:47:51 +00:00
|
|
|
self._ws_send_command("claim", nameplate=self._nameplate_id)
|
2016-05-24 05:53:00 +00:00
|
|
|
self._nameplate_state = OPEN
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _response_handle_claimed(self, msg):
|
2016-05-21 01:49:20 +00:00
|
|
|
mailbox_id = msg["mailbox"]
|
2016-06-04 19:47:51 +00:00
|
|
|
assert isinstance(mailbox_id, type("")), type(mailbox_id)
|
2016-05-21 01:49:20 +00:00
|
|
|
self._mailbox_id = mailbox_id
|
|
|
|
self._event_learned_mailbox()
|
|
|
|
|
|
|
|
def _event_learned_mailbox(self):
|
2016-06-22 08:04:05 +00:00
|
|
|
if not self._mailbox_id: raise InternalError
|
2016-05-24 05:53:00 +00:00
|
|
|
assert self._mailbox_state == CLOSED, self._mailbox_state
|
|
|
|
if self._closing:
|
|
|
|
return
|
2016-06-04 19:47:51 +00:00
|
|
|
self._ws_send_command("open", mailbox=self._mailbox_id)
|
2016-05-24 05:53:00 +00:00
|
|
|
self._mailbox_state = OPEN
|
2016-05-21 01:49:20 +00:00
|
|
|
# causes old messages to be sent now, and subscribes to new messages
|
|
|
|
self._maybe_send_pake()
|
|
|
|
self._maybe_send_phase_messages()
|
|
|
|
|
|
|
|
def _maybe_send_pake(self):
|
|
|
|
# TODO: deal with reentrant call
|
2016-05-24 05:53:00 +00:00
|
|
|
if not (self._connection_state == OPEN
|
|
|
|
and self._mailbox_state == OPEN
|
2016-05-21 01:49:20 +00:00
|
|
|
and self._flag_need_to_send_PAKE):
|
|
|
|
return
|
2016-06-04 19:47:51 +00:00
|
|
|
body = {"pake_v1": bytes_to_hexstr(self._msg1)}
|
2016-05-29 01:30:36 +00:00
|
|
|
payload = dict_to_bytes(body)
|
2016-06-04 19:47:51 +00:00
|
|
|
self._msg_send("pake", payload)
|
2016-05-23 01:40:44 +00:00
|
|
|
self._flag_need_to_send_PAKE = False
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
def _event_received_pake(self, pake_msg):
|
2016-05-29 01:30:36 +00:00
|
|
|
payload = bytes_to_dict(pake_msg)
|
2016-06-04 19:47:51 +00:00
|
|
|
msg2 = hexstr_to_bytes(payload["pake_v1"])
|
2016-05-21 01:49:20 +00:00
|
|
|
with self._timing.add("pake2", waiting="crypto"):
|
2016-05-29 01:30:36 +00:00
|
|
|
self._key = self._sp.finish(msg2)
|
2016-05-21 01:49:20 +00:00
|
|
|
self._event_established_key()
|
|
|
|
|
|
|
|
def _event_established_key(self):
|
|
|
|
self._timing.add("key established")
|
2016-06-05 06:21:44 +00:00
|
|
|
self._maybe_notify_key()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
# both sides send different (random) version messages
|
|
|
|
self._send_version_message()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-24 20:31:03 +00:00
|
|
|
verifier = self._derive_key(b"wormhole:verifier")
|
2016-05-21 01:49:20 +00:00
|
|
|
self._event_computed_verifier(verifier)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
self._maybe_check_version()
|
2016-05-22 18:31:00 +00:00
|
|
|
self._maybe_send_phase_messages()
|
|
|
|
|
2016-06-05 06:21:44 +00:00
|
|
|
def _API_establish_key(self):
|
|
|
|
if self._error: return defer.fail(self._error)
|
2016-12-16 09:33:17 +00:00
|
|
|
if self._establish_key_called: raise InternalError
|
|
|
|
self._establish_key_called = True
|
2016-06-05 10:10:32 +00:00
|
|
|
if self._key is not None:
|
2016-06-05 06:21:44 +00:00
|
|
|
return defer.succeed(True)
|
|
|
|
self._key_waiter = defer.Deferred()
|
|
|
|
return self._key_waiter
|
|
|
|
|
|
|
|
def _maybe_notify_key(self):
|
|
|
|
if self._key is None:
|
|
|
|
return
|
|
|
|
if self._error:
|
|
|
|
result = failure.Failure(self._error)
|
|
|
|
else:
|
|
|
|
result = True
|
|
|
|
if self._key_waiter and not self._key_waiter.called:
|
|
|
|
self._key_waiter.callback(result)
|
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
def _send_version_message(self):
|
2016-05-26 01:27:37 +00:00
|
|
|
# this is encrypted like a normal phase message, and includes a
|
|
|
|
# dictionary of version flags to let the other Wormhole know what
|
|
|
|
# we're capable of (for future expansion)
|
2016-05-29 01:19:45 +00:00
|
|
|
plaintext = dict_to_bytes(self._my_versions)
|
2016-06-04 19:47:51 +00:00
|
|
|
phase = "version"
|
2016-05-26 01:27:37 +00:00
|
|
|
data_key = self._derive_phase_key(self._side, phase)
|
|
|
|
encrypted = self._encrypt_data(data_key, plaintext)
|
|
|
|
self._msg_send(phase, encrypted)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_verify(self):
|
2016-05-23 07:14:39 +00:00
|
|
|
if self._error: return defer.fail(self._error)
|
2016-06-22 08:04:05 +00:00
|
|
|
if self._get_verifier_called: raise InternalError
|
2016-05-23 01:40:44 +00:00
|
|
|
self._get_verifier_called = True
|
2016-05-26 01:05:02 +00:00
|
|
|
if self._verify_result:
|
|
|
|
return defer.succeed(self._verify_result) # bytes or Failure
|
2016-05-24 05:53:00 +00:00
|
|
|
self._verifier_waiter = defer.Deferred()
|
2016-05-23 01:40:44 +00:00
|
|
|
return self._verifier_waiter
|
|
|
|
|
2016-05-21 01:49:20 +00:00
|
|
|
def _event_computed_verifier(self, verifier):
|
2016-05-24 05:53:00 +00:00
|
|
|
self._verifier = verifier
|
2016-05-26 01:05:02 +00:00
|
|
|
self._maybe_notify_verify()
|
|
|
|
|
|
|
|
def _maybe_notify_verify(self):
|
2016-05-26 02:13:37 +00:00
|
|
|
if not (self._verifier and self._version_checked):
|
2016-05-26 01:05:02 +00:00
|
|
|
return
|
|
|
|
if self._error:
|
|
|
|
self._verify_result = failure.Failure(self._error)
|
|
|
|
else:
|
|
|
|
self._verify_result = self._verifier
|
|
|
|
if self._verifier_waiter and not self._verifier_waiter.called:
|
|
|
|
self._verifier_waiter.callback(self._verify_result)
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
def _event_received_version(self, side, body):
|
2016-05-26 01:05:02 +00:00
|
|
|
# We ought to have the master key by now, because sensible peers
|
2016-05-26 02:09:00 +00:00
|
|
|
# should always send "pake" before sending "version". It might be
|
2016-05-26 01:05:02 +00:00
|
|
|
# nice to relax this requirement, which means storing the received
|
2016-05-26 02:13:37 +00:00
|
|
|
# version message, and having _event_established_key call
|
|
|
|
# _check_version()
|
|
|
|
self._version_message = (side, body)
|
|
|
|
self._maybe_check_version()
|
2016-05-26 01:05:02 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
def _maybe_check_version(self):
|
|
|
|
if not (self._key and self._version_message):
|
2016-05-26 01:05:02 +00:00
|
|
|
return
|
2016-05-26 02:13:37 +00:00
|
|
|
if self._version_checked:
|
2016-05-26 01:05:02 +00:00
|
|
|
return
|
2016-05-26 02:13:37 +00:00
|
|
|
self._version_checked = True
|
2016-05-26 01:27:37 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
side, body = self._version_message
|
2016-06-04 19:47:51 +00:00
|
|
|
data_key = self._derive_phase_key(side, "version")
|
2016-05-26 01:27:37 +00:00
|
|
|
try:
|
|
|
|
plaintext = self._decrypt_data(data_key, body)
|
|
|
|
except CryptoError:
|
2016-05-21 01:49:20 +00:00
|
|
|
# this makes all API calls fail
|
2016-05-23 07:14:39 +00:00
|
|
|
if self.DEBUG: print("CONFIRM FAILED")
|
2016-06-04 19:47:51 +00:00
|
|
|
self._signal_error(WrongPasswordError(), "scary")
|
2016-05-26 01:27:37 +00:00
|
|
|
return
|
2016-05-29 01:19:45 +00:00
|
|
|
msg = bytes_to_dict(plaintext)
|
2016-05-26 01:27:37 +00:00
|
|
|
self._version_received(msg)
|
|
|
|
|
2016-05-26 01:05:02 +00:00
|
|
|
self._maybe_notify_verify()
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-26 01:27:37 +00:00
|
|
|
def _version_received(self, msg):
|
|
|
|
self._their_versions = msg
|
2016-05-22 18:31:00 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_send(self, outbound_data):
|
2016-05-23 07:14:39 +00:00
|
|
|
if self._error: raise self._error
|
2016-05-22 18:31:00 +00:00
|
|
|
if not isinstance(outbound_data, type(b"")):
|
|
|
|
raise TypeError(type(outbound_data))
|
|
|
|
phase = self._next_send_phase
|
|
|
|
self._next_send_phase += 1
|
2016-05-23 01:40:44 +00:00
|
|
|
self._plaintext_to_send.append( (phase, outbound_data) )
|
|
|
|
with self._timing.add("API send", phase=phase):
|
2016-05-22 18:31:00 +00:00
|
|
|
self._maybe_send_phase_messages()
|
|
|
|
|
2016-05-24 20:47:15 +00:00
|
|
|
def _derive_phase_key(self, side, phase):
|
2016-06-04 19:47:51 +00:00
|
|
|
assert isinstance(side, type("")), type(side)
|
|
|
|
assert isinstance(phase, type("")), type(phase)
|
2016-05-24 20:47:15 +00:00
|
|
|
side_bytes = side.encode("ascii")
|
|
|
|
phase_bytes = phase.encode("ascii")
|
|
|
|
purpose = (b"wormhole:phase:"
|
|
|
|
+ sha256(side_bytes).digest()
|
|
|
|
+ sha256(phase_bytes).digest())
|
2016-05-24 20:31:03 +00:00
|
|
|
return self._derive_key(purpose)
|
2016-05-24 20:26:08 +00:00
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def _maybe_send_phase_messages(self):
|
|
|
|
# TODO: deal with reentrant call
|
2016-05-24 05:53:00 +00:00
|
|
|
if not (self._connection_state == OPEN
|
|
|
|
and self._mailbox_state == OPEN
|
|
|
|
and self._key):
|
2016-05-22 18:31:00 +00:00
|
|
|
return
|
|
|
|
plaintexts = self._plaintext_to_send
|
|
|
|
self._plaintext_to_send = []
|
|
|
|
for pm in plaintexts:
|
2016-05-24 20:47:15 +00:00
|
|
|
(phase_int, plaintext) = pm
|
|
|
|
assert isinstance(phase_int, int), type(phase_int)
|
2016-06-04 19:47:51 +00:00
|
|
|
phase = "%d" % phase_int
|
2016-05-24 20:47:15 +00:00
|
|
|
data_key = self._derive_phase_key(self._side, phase)
|
2016-05-22 18:31:00 +00:00
|
|
|
encrypted = self._encrypt_data(data_key, plaintext)
|
2016-05-24 20:47:15 +00:00
|
|
|
self._msg_send(phase, encrypted)
|
2016-05-22 18:31:00 +00:00
|
|
|
|
|
|
|
def _encrypt_data(self, key, data):
|
|
|
|
# Without predefined roles, we can't derive predictably unique keys
|
|
|
|
# for each side, so we use the same key for both. We use random
|
|
|
|
# nonces to keep the messages distinct, and we automatically ignore
|
|
|
|
# reflections.
|
2016-05-23 01:40:44 +00:00
|
|
|
# TODO: HKDF(side, nonce, key) ?? include 'side' to prevent
|
|
|
|
# reflections, since we no longer compare messages
|
2016-05-22 18:31:00 +00:00
|
|
|
assert isinstance(key, type(b"")), type(key)
|
|
|
|
assert isinstance(data, type(b"")), type(data)
|
|
|
|
assert len(key) == SecretBox.KEY_SIZE, len(key)
|
|
|
|
box = SecretBox(key)
|
|
|
|
nonce = utils.random(SecretBox.NONCE_SIZE)
|
|
|
|
return box.encrypt(data, nonce)
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
def _msg_send(self, phase, body):
|
2016-06-22 08:04:05 +00:00
|
|
|
if phase in self._sent_phases: raise InternalError
|
2016-05-24 05:53:00 +00:00
|
|
|
assert self._mailbox_state == OPEN, self._mailbox_state
|
2016-05-23 01:40:44 +00:00
|
|
|
self._sent_phases.add(phase)
|
2016-05-22 18:31:00 +00:00
|
|
|
# TODO: retry on failure, with exponential backoff. We're guarding
|
|
|
|
# against the rendezvous server being temporarily offline.
|
2016-05-23 01:40:44 +00:00
|
|
|
self._timing.add("add", phase=phase)
|
2016-06-04 19:47:51 +00:00
|
|
|
self._ws_send_command("add", phase=phase, body=bytes_to_hexstr(body))
|
2016-05-22 18:31:00 +00:00
|
|
|
|
|
|
|
def _event_mailbox_used(self):
|
2016-05-23 01:40:44 +00:00
|
|
|
if self.DEBUG: print("_event_mailbox_used")
|
2016-05-22 18:31:00 +00:00
|
|
|
if self._flag_need_to_see_mailbox_used:
|
2016-05-23 01:40:44 +00:00
|
|
|
self._maybe_release_nameplate()
|
2016-05-22 18:31:00 +00:00
|
|
|
self._flag_need_to_see_mailbox_used = False
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_derive_key(self, purpose, length):
|
2016-05-23 07:14:39 +00:00
|
|
|
if self._error: raise self._error
|
2016-05-25 23:29:56 +00:00
|
|
|
if self._key is None:
|
2016-06-22 08:04:05 +00:00
|
|
|
raise InternalError # call derive_key after get_verifier() or get()
|
2016-06-04 19:47:51 +00:00
|
|
|
if not isinstance(purpose, type("")): raise TypeError(type(purpose))
|
2016-05-24 20:31:03 +00:00
|
|
|
return self._derive_key(to_bytes(purpose), length)
|
2016-05-24 05:53:00 +00:00
|
|
|
|
|
|
|
def _derive_key(self, purpose, length=SecretBox.KEY_SIZE):
|
2016-05-24 20:31:03 +00:00
|
|
|
if not isinstance(purpose, type(b"")): raise TypeError(type(purpose))
|
2016-05-22 18:31:00 +00:00
|
|
|
if self._key is None:
|
2016-06-22 08:04:05 +00:00
|
|
|
raise InternalError # call derive_key after get_verifier() or get()
|
2016-05-24 20:31:03 +00:00
|
|
|
return HKDF(self._key, length, CTXinfo=purpose)
|
2016-05-22 18:31:00 +00:00
|
|
|
|
|
|
|
def _response_handle_message(self, msg):
|
2016-05-21 01:49:20 +00:00
|
|
|
side = msg["side"]
|
|
|
|
phase = msg["phase"]
|
2016-06-04 19:47:51 +00:00
|
|
|
assert isinstance(phase, type("")), type(phase)
|
2016-05-29 01:19:45 +00:00
|
|
|
body = hexstr_to_bytes(msg["body"])
|
2016-05-21 01:49:20 +00:00
|
|
|
if side == self._side:
|
|
|
|
return
|
2016-05-24 20:47:15 +00:00
|
|
|
self._event_received_peer_message(side, phase, body)
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 20:47:15 +00:00
|
|
|
def _event_received_peer_message(self, side, phase, body):
|
2016-05-21 01:49:20 +00:00
|
|
|
# any message in the mailbox means we no longer need the nameplate
|
|
|
|
self._event_mailbox_used()
|
2016-05-24 23:15:43 +00:00
|
|
|
|
2016-05-25 23:29:56 +00:00
|
|
|
if self._closing:
|
|
|
|
log.msg("received peer message while closing '%s'" % phase)
|
2016-12-26 20:27:14 +00:00
|
|
|
if phase in self._received_messages:
|
|
|
|
log.msg("ignoring duplicate peer message '%s'" % phase)
|
|
|
|
return
|
2016-05-25 23:29:56 +00:00
|
|
|
|
2016-06-04 19:47:51 +00:00
|
|
|
if phase == "pake":
|
2016-12-26 20:27:14 +00:00
|
|
|
self._received_messages["pake"] = body
|
2016-05-24 23:15:43 +00:00
|
|
|
return self._event_received_pake(body)
|
2016-06-04 19:47:51 +00:00
|
|
|
if phase == "version":
|
2016-12-26 20:27:14 +00:00
|
|
|
self._received_messages["version"] = body
|
2016-05-26 02:13:37 +00:00
|
|
|
return self._event_received_version(side, body)
|
2016-05-24 23:15:43 +00:00
|
|
|
if re.search(r'^\d+$', phase):
|
|
|
|
return self._event_received_phase_message(side, phase, body)
|
|
|
|
# ignore unrecognized phases, for forwards-compatibility
|
|
|
|
log.msg("received unknown phase '%s'" % phase)
|
|
|
|
|
|
|
|
def _event_received_phase_message(self, side, phase, body):
|
|
|
|
# It's a numbered phase message, aimed at the application above us.
|
|
|
|
# Decrypt and deliver upstairs, notifying anyone waiting on it
|
2016-05-22 18:31:00 +00:00
|
|
|
try:
|
2016-05-24 20:47:15 +00:00
|
|
|
data_key = self._derive_phase_key(side, phase)
|
2016-05-23 01:40:44 +00:00
|
|
|
plaintext = self._decrypt_data(data_key, body)
|
2016-05-22 18:31:00 +00:00
|
|
|
except CryptoError:
|
2016-05-24 05:53:00 +00:00
|
|
|
e = WrongPasswordError()
|
2016-06-04 19:47:51 +00:00
|
|
|
self._signal_error(e, "scary") # flunk all other API calls
|
2016-05-24 05:53:00 +00:00
|
|
|
# make tests fail, if they aren't explicitly catching it
|
|
|
|
if self.DEBUG: print("CryptoError in msg received")
|
|
|
|
log.err(e)
|
|
|
|
if self.DEBUG: print(" did log.err", e)
|
|
|
|
return # ignore this message
|
2016-05-23 01:40:44 +00:00
|
|
|
self._received_messages[phase] = plaintext
|
2016-05-22 18:31:00 +00:00
|
|
|
if phase in self._receive_waiters:
|
|
|
|
d = self._receive_waiters.pop(phase)
|
2016-05-23 01:40:44 +00:00
|
|
|
d.callback(plaintext)
|
2016-05-22 18:31:00 +00:00
|
|
|
|
|
|
|
def _decrypt_data(self, key, encrypted):
|
|
|
|
assert isinstance(key, type(b"")), type(key)
|
|
|
|
assert isinstance(encrypted, type(b"")), type(encrypted)
|
|
|
|
assert len(key) == SecretBox.KEY_SIZE, len(key)
|
|
|
|
box = SecretBox(key)
|
|
|
|
data = box.decrypt(encrypted)
|
|
|
|
return data
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _API_get(self):
|
2016-05-23 07:14:39 +00:00
|
|
|
if self._error: return defer.fail(self._error)
|
2016-06-04 19:47:51 +00:00
|
|
|
phase = "%d" % self._next_receive_phase
|
2016-05-22 18:31:00 +00:00
|
|
|
self._next_receive_phase += 1
|
|
|
|
with self._timing.add("API get", phase=phase):
|
2016-05-23 01:40:44 +00:00
|
|
|
if phase in self._received_messages:
|
|
|
|
return defer.succeed(self._received_messages[phase])
|
2016-05-22 18:31:00 +00:00
|
|
|
d = self._receive_waiters[phase] = defer.Deferred()
|
2016-05-23 01:40:44 +00:00
|
|
|
return d
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _signal_error(self, error, mood):
|
|
|
|
if self.DEBUG: print("_signal_error", error, mood)
|
|
|
|
if self._error:
|
2016-05-23 07:14:39 +00:00
|
|
|
return
|
2016-05-24 05:53:00 +00:00
|
|
|
self._maybe_close(error, mood)
|
|
|
|
if self.DEBUG: print("_signal_error done")
|
2016-05-23 07:14:39 +00:00
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
@inlineCallbacks
|
2016-06-04 19:47:51 +00:00
|
|
|
def _API_close(self, res, mood="happy"):
|
2016-05-24 06:59:49 +00:00
|
|
|
if self.DEBUG: print("close")
|
2016-06-22 08:04:05 +00:00
|
|
|
if self._close_called: raise InternalError
|
2016-05-24 05:53:00 +00:00
|
|
|
self._close_called = True
|
|
|
|
self._maybe_close(WormholeClosedError(), mood)
|
2016-05-24 06:59:49 +00:00
|
|
|
if self.DEBUG: print("waiting for disconnect")
|
|
|
|
yield self._disconnect_waiter
|
|
|
|
returnValue(res)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _maybe_close(self, error, mood):
|
|
|
|
if self._closing:
|
|
|
|
return
|
|
|
|
|
|
|
|
# ordering constraints:
|
|
|
|
# * must wait for nameplate/mailbox acks before closing the websocket
|
|
|
|
# * must mark APIs for failure before errbacking Deferreds
|
|
|
|
# * since we give up control
|
|
|
|
# * must mark self._closing before errbacking Deferreds
|
|
|
|
# * since caller may call close() when we give up control
|
|
|
|
# * and close() will reenter _maybe_close
|
|
|
|
|
|
|
|
self._error = error # causes new API calls to fail
|
|
|
|
|
|
|
|
# since we're about to give up control by errbacking any API
|
|
|
|
# Deferreds, set self._closing, to make sure that a new call to
|
|
|
|
# close() isn't going to confuse anything
|
|
|
|
self._closing = True
|
|
|
|
|
|
|
|
# now errback all API deferreds except close(): get_code,
|
|
|
|
# input_code, verify, get
|
2016-05-26 22:38:19 +00:00
|
|
|
if self._input_code_waiter and not self._input_code_waiter.called:
|
|
|
|
self._input_code_waiter.errback(error)
|
2016-05-24 05:53:00 +00:00
|
|
|
for d in self._connection_waiters: # input_code, get_code (early)
|
|
|
|
if self.DEBUG: print("EB cw")
|
|
|
|
d.errback(error)
|
|
|
|
if self._get_code: # get_code (late)
|
|
|
|
if self.DEBUG: print("EB gc")
|
|
|
|
self._get_code._allocated_d.errback(error)
|
|
|
|
if self._verifier_waiter and not self._verifier_waiter.called:
|
|
|
|
if self.DEBUG: print("EB VW")
|
|
|
|
self._verifier_waiter.errback(error)
|
2016-06-05 09:55:31 +00:00
|
|
|
if self._key_waiter and not self._key_waiter.called:
|
|
|
|
if self.DEBUG: print("EB KW")
|
|
|
|
self._key_waiter.errback(error)
|
2016-05-24 05:53:00 +00:00
|
|
|
for d in self._receive_waiters.values():
|
|
|
|
if self.DEBUG: print("EB RW")
|
|
|
|
d.errback(error)
|
|
|
|
# Release nameplate and close mailbox, if either was claimed/open.
|
|
|
|
# Since _closing is True when both ACKs come back, the handlers will
|
|
|
|
# close the websocket. When *that* finishes, _disconnect_waiter()
|
|
|
|
# will fire.
|
|
|
|
self._maybe_release_nameplate()
|
|
|
|
self._maybe_close_mailbox(mood)
|
|
|
|
# In the off chance we got closed before we even claimed the
|
|
|
|
# nameplate, give _maybe_finished_closing a chance to run now.
|
|
|
|
self._maybe_finished_closing()
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
def _maybe_release_nameplate(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
if self.DEBUG: print("_maybe_release_nameplate", self._nameplate_state)
|
|
|
|
if self._nameplate_state == OPEN:
|
2016-05-23 01:40:44 +00:00
|
|
|
if self.DEBUG: print(" sending release")
|
2016-06-04 19:47:51 +00:00
|
|
|
self._ws_send_command("release")
|
2016-05-24 05:53:00 +00:00
|
|
|
self._nameplate_state = CLOSING
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
def _response_handle_released(self, msg):
|
2016-05-24 05:53:00 +00:00
|
|
|
self._nameplate_state = CLOSED
|
|
|
|
self._maybe_finished_closing()
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def _maybe_close_mailbox(self, mood):
|
|
|
|
if self.DEBUG: print("_maybe_close_mailbox", self._mailbox_state)
|
|
|
|
if self._mailbox_state == OPEN:
|
2016-05-23 07:14:39 +00:00
|
|
|
if self.DEBUG: print(" sending close")
|
2016-06-04 19:47:51 +00:00
|
|
|
self._ws_send_command("close", mood=mood)
|
2016-05-24 05:53:00 +00:00
|
|
|
self._mailbox_state = CLOSING
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
def _response_handle_closed(self, msg):
|
2016-05-24 05:53:00 +00:00
|
|
|
self._mailbox_state = CLOSED
|
|
|
|
self._maybe_finished_closing()
|
|
|
|
|
|
|
|
def _maybe_finished_closing(self):
|
|
|
|
if self.DEBUG: print("_maybe_finished_closing", self._closing, self._nameplate_state, self._mailbox_state, self._connection_state)
|
|
|
|
if not self._closing:
|
|
|
|
return
|
|
|
|
if (self._nameplate_state == CLOSED
|
|
|
|
and self._mailbox_state == CLOSED
|
|
|
|
and self._connection_state == OPEN):
|
|
|
|
self._connection_state = CLOSING
|
|
|
|
self._drop_connection()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
def _drop_connection(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
# separate method so it can be overridden by tests
|
|
|
|
self._ws.transport.loseConnection() # probably flushes output
|
2016-05-23 01:40:44 +00:00
|
|
|
# calls _ws_closed() when done
|
|
|
|
|
|
|
|
def _ws_closed(self, wasClean, code, reason):
|
2016-05-24 05:53:00 +00:00
|
|
|
# For now (until we add reconnection), losing the websocket means
|
|
|
|
# losing everything. Make all API callers fail. Help someone waiting
|
|
|
|
# in close() to finish
|
|
|
|
self._connection_state = CLOSED
|
2016-05-23 01:45:50 +00:00
|
|
|
self._disconnect_waiter.callback(None)
|
2016-05-24 05:53:00 +00:00
|
|
|
self._maybe_finished_closing()
|
|
|
|
|
|
|
|
# what needs to happen when _ws_closed() happens unexpectedly
|
|
|
|
# * errback all API deferreds
|
|
|
|
# * maybe: cause new API calls to fail
|
|
|
|
# * obviously can't release nameplate or close mailbox
|
|
|
|
# * can't re-close websocket
|
|
|
|
# * close(wait=True) callers should fire right away
|
2016-05-21 01:49:20 +00:00
|
|
|
|
2017-01-06 16:25:32 +00:00
|
|
|
def wormhole(appid, relay_url, reactor, tor_manager=None, timing=None,
|
|
|
|
stderr=sys.stderr):
|
2016-05-23 01:40:44 +00:00
|
|
|
timing = timing or DebugTiming()
|
2017-01-06 16:25:32 +00:00
|
|
|
w = _Wormhole(appid, relay_url, reactor, tor_manager, timing, stderr)
|
2016-05-21 01:49:20 +00:00
|
|
|
w._start()
|
|
|
|
return w
|
|
|
|
|
2016-05-28 01:47:14 +00:00
|
|
|
#def wormhole_from_serialized(data, reactor, timing=None):
|
|
|
|
# timing = timing or DebugTiming()
|
|
|
|
# w = _Wormhole.from_serialized(data, reactor, timing)
|
|
|
|
# return w
|