2016-05-22 18:31:00 +00:00
|
|
|
from __future__ import print_function
|
2016-05-23 07:14:39 +00:00
|
|
|
import os, json, re, gc
|
2016-05-23 01:40:44 +00:00
|
|
|
from binascii import hexlify, unhexlify
|
2016-05-22 18:31:00 +00:00
|
|
|
import mock
|
|
|
|
from twisted.trial import unittest
|
|
|
|
from twisted.internet import reactor
|
2016-05-23 01:40:44 +00:00
|
|
|
from twisted.internet.defer import Deferred, gatherResults, inlineCallbacks
|
|
|
|
from .common import ServerBase
|
|
|
|
from .. import wormhole
|
2016-05-24 05:53:00 +00:00
|
|
|
from ..errors import WrongPasswordError, WelcomeError, UsageError
|
2016-05-23 01:40:44 +00:00
|
|
|
from spake2 import SPAKE2_Symmetric
|
2016-05-22 18:31:00 +00:00
|
|
|
from ..timing import DebugTiming
|
2016-05-29 01:30:36 +00:00
|
|
|
from ..util import (bytes_to_dict, dict_to_bytes,
|
|
|
|
hexstr_to_bytes, bytes_to_hexstr)
|
2016-05-24 05:53:00 +00:00
|
|
|
from nacl.secret import SecretBox
|
2016-05-22 18:31:00 +00:00
|
|
|
|
|
|
|
APPID = u"appid"
|
|
|
|
|
|
|
|
class MockWebSocket:
|
|
|
|
def __init__(self):
|
|
|
|
self._payloads = []
|
|
|
|
def sendMessage(self, payload, is_binary):
|
|
|
|
assert not is_binary
|
|
|
|
self._payloads.append(payload)
|
|
|
|
|
|
|
|
def outbound(self):
|
|
|
|
out = []
|
|
|
|
while self._payloads:
|
|
|
|
p = self._payloads.pop(0)
|
|
|
|
out.append(json.loads(p.decode("utf-8")))
|
|
|
|
return out
|
|
|
|
|
|
|
|
def response(w, **kwargs):
|
|
|
|
payload = json.dumps(kwargs).encode("utf-8")
|
|
|
|
w._ws_dispatch_response(payload)
|
|
|
|
|
|
|
|
class Welcome(unittest.TestCase):
|
2016-05-23 01:40:44 +00:00
|
|
|
def test_tolerate_no_current_version(self):
|
|
|
|
w = wormhole._WelcomeHandler(u"relay_url", u"current_version", None)
|
2016-05-22 18:31:00 +00:00
|
|
|
w.handle_welcome({})
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
def test_print_motd(self):
|
|
|
|
w = wormhole._WelcomeHandler(u"relay_url", u"current_version", None)
|
|
|
|
with mock.patch("sys.stderr") as stderr:
|
|
|
|
w.handle_welcome({u"motd": u"message of\nthe day"})
|
|
|
|
self.assertEqual(stderr.method_calls,
|
|
|
|
[mock.call.write(u"Server (at relay_url) says:\n"
|
|
|
|
" message of\n the day"),
|
|
|
|
mock.call.write(u"\n")])
|
|
|
|
# motd is only displayed once
|
|
|
|
with mock.patch("sys.stderr") as stderr2:
|
|
|
|
w.handle_welcome({u"motd": u"second message"})
|
|
|
|
self.assertEqual(stderr2.method_calls, [])
|
|
|
|
|
|
|
|
def test_current_version(self):
|
|
|
|
w = wormhole._WelcomeHandler(u"relay_url", u"2.0", None)
|
|
|
|
with mock.patch("sys.stderr") as stderr:
|
|
|
|
w.handle_welcome({u"current_version": u"2.0"})
|
|
|
|
self.assertEqual(stderr.method_calls, [])
|
|
|
|
|
|
|
|
with mock.patch("sys.stderr") as stderr:
|
|
|
|
w.handle_welcome({u"current_version": u"3.0"})
|
|
|
|
exp1 = (u"Warning: errors may occur unless both sides are"
|
|
|
|
" running the same version")
|
|
|
|
exp2 = (u"Server claims 3.0 is current, but ours is 2.0")
|
|
|
|
self.assertEqual(stderr.method_calls,
|
|
|
|
[mock.call.write(exp1),
|
|
|
|
mock.call.write(u"\n"),
|
|
|
|
mock.call.write(exp2),
|
|
|
|
mock.call.write(u"\n"),
|
|
|
|
])
|
|
|
|
|
|
|
|
# warning is only displayed once
|
|
|
|
with mock.patch("sys.stderr") as stderr:
|
|
|
|
w.handle_welcome({u"current_version": u"3.0"})
|
|
|
|
self.assertEqual(stderr.method_calls, [])
|
|
|
|
|
|
|
|
def test_non_release_version(self):
|
|
|
|
w = wormhole._WelcomeHandler(u"relay_url", u"2.0-dirty", None)
|
|
|
|
with mock.patch("sys.stderr") as stderr:
|
|
|
|
w.handle_welcome({u"current_version": u"3.0"})
|
|
|
|
self.assertEqual(stderr.method_calls, [])
|
|
|
|
|
|
|
|
def test_signal_error(self):
|
|
|
|
se = mock.Mock()
|
|
|
|
w = wormhole._WelcomeHandler(u"relay_url", u"2.0", se)
|
|
|
|
w.handle_welcome({})
|
|
|
|
self.assertEqual(se.mock_calls, [])
|
|
|
|
|
|
|
|
w.handle_welcome({u"error": u"oops"})
|
2016-05-23 07:14:39 +00:00
|
|
|
self.assertEqual(len(se.mock_calls), 1)
|
2016-05-26 22:37:24 +00:00
|
|
|
self.assertEqual(len(se.mock_calls[0][1]), 2) # posargs
|
2016-05-23 07:14:39 +00:00
|
|
|
we = se.mock_calls[0][1][0]
|
2016-05-24 05:53:00 +00:00
|
|
|
self.assertIsInstance(we, WelcomeError)
|
2016-05-23 07:14:39 +00:00
|
|
|
self.assertEqual(we.args, (u"oops",))
|
2016-05-26 22:37:24 +00:00
|
|
|
mood = se.mock_calls[0][1][1]
|
|
|
|
self.assertEqual(mood, u"unwelcome")
|
2016-05-23 07:14:39 +00:00
|
|
|
# alas WelcomeError instances don't compare against each other
|
2016-05-24 05:53:00 +00:00
|
|
|
#self.assertEqual(se.mock_calls, [mock.call(WelcomeError(u"oops"))])
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
class InputCode(unittest.TestCase):
|
|
|
|
def test_list(self):
|
|
|
|
send_command = mock.Mock()
|
|
|
|
ic = wormhole._InputCode(None, u"prompt", 2, send_command,
|
|
|
|
DebugTiming())
|
|
|
|
d = ic._list()
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(send_command.mock_calls, [mock.call(u"list")])
|
|
|
|
ic._response_handle_nameplates({u"type": u"nameplates",
|
|
|
|
u"nameplates": [{u"id": u"123"}]})
|
|
|
|
res = self.successResultOf(d)
|
|
|
|
self.assertEqual(res, [u"123"])
|
|
|
|
|
|
|
|
class GetCode(unittest.TestCase):
|
|
|
|
def test_get(self):
|
|
|
|
send_command = mock.Mock()
|
|
|
|
gc = wormhole._GetCode(2, send_command, DebugTiming())
|
|
|
|
d = gc.go()
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(send_command.mock_calls, [mock.call(u"allocate")])
|
|
|
|
# TODO: nameplate attributes get added and checked here
|
|
|
|
gc._response_handle_allocated({u"type": u"allocated",
|
|
|
|
u"nameplate": u"123"})
|
|
|
|
code = self.successResultOf(d)
|
|
|
|
self.assertIsInstance(code, type(u""))
|
|
|
|
self.assert_(code.startswith(u"123-"))
|
|
|
|
pieces = code.split(u"-")
|
|
|
|
self.assertEqual(len(pieces), 3) # nameplate plus two words
|
|
|
|
self.assert_(re.search(r'^\d+-\w+-\w+$', code), code)
|
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
class Basic(unittest.TestCase):
|
2016-05-23 07:14:39 +00:00
|
|
|
def tearDown(self):
|
|
|
|
# flush out any errorful Deferreds left dangling in cycles
|
|
|
|
gc.collect()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
def check_out(self, out, **kwargs):
|
|
|
|
# Assert that each kwarg is present in the 'out' dict. Ignore other
|
|
|
|
# keys ('msgid' in particular)
|
|
|
|
for key, value in kwargs.items():
|
|
|
|
self.assertIn(key, out)
|
|
|
|
self.assertEqual(out[key], value, (out, key, value))
|
|
|
|
|
|
|
|
def check_outbound(self, ws, types):
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), len(types), (out, types))
|
|
|
|
for i,t in enumerate(types):
|
|
|
|
self.assertEqual(out[i][u"type"], t, (i,t,out))
|
|
|
|
return out
|
2016-05-22 18:31:00 +00:00
|
|
|
|
2016-05-23 07:14:39 +00:00
|
|
|
def make_pake(self, code, side, msg1):
|
|
|
|
sp2 = SPAKE2_Symmetric(wormhole.to_bytes(code),
|
|
|
|
idSymmetric=wormhole.to_bytes(APPID))
|
|
|
|
msg2 = sp2.start()
|
|
|
|
key = sp2.finish(msg1)
|
2016-05-29 01:30:36 +00:00
|
|
|
return key, msg2
|
2016-05-23 07:14:39 +00:00
|
|
|
|
|
|
|
def test_create(self):
|
|
|
|
wormhole._Wormhole(APPID, u"relay_url", reactor, None, None)
|
|
|
|
|
2016-05-22 18:31:00 +00:00
|
|
|
def test_basic(self):
|
|
|
|
# We don't call w._start(), so this doesn't create a WebSocket
|
2016-05-23 01:40:44 +00:00
|
|
|
# connection. We provide a mock connection instead. If we wanted to
|
|
|
|
# exercise _connect, we'd mock out WSFactory.
|
2016-05-22 18:31:00 +00:00
|
|
|
# w._connect = lambda self: None
|
|
|
|
# w._event_connected(mock_ws)
|
|
|
|
# w._event_ws_opened()
|
|
|
|
# w._ws_dispatch_response(payload)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
timing = DebugTiming()
|
|
|
|
with mock.patch("wormhole.wormhole._WelcomeHandler") as wh_c:
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
wh = wh_c.return_value
|
2016-05-22 18:31:00 +00:00
|
|
|
self.assertEqual(w._ws_url, u"relay_url")
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertTrue(w._flag_need_nameplate)
|
|
|
|
self.assertTrue(w._flag_need_to_build_msg1)
|
|
|
|
self.assertTrue(w._flag_need_to_send_PAKE)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
v = w.verify()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
w._drop_connection = mock.Mock()
|
2016-05-22 18:31:00 +00:00
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 0)
|
|
|
|
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 1)
|
2016-05-23 01:40:44 +00:00
|
|
|
self.check_out(out[0], type=u"bind", appid=APPID, side=w._side)
|
2016-05-22 18:31:00 +00:00
|
|
|
self.assertIn(u"id", out[0])
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
# WelcomeHandler should get called upon 'welcome' response. Its full
|
|
|
|
# behavior is exercised in 'Welcome' above.
|
2016-05-22 18:31:00 +00:00
|
|
|
WELCOME = {u"foo": u"bar"}
|
|
|
|
response(w, type="welcome", welcome=WELCOME)
|
|
|
|
self.assertEqual(wh.mock_calls, [mock.call.handle_welcome(WELCOME)])
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
# because we're connected, setting the code also claims the mailbox
|
|
|
|
CODE = u"123-foo-bar"
|
|
|
|
w.set_code(CODE)
|
|
|
|
self.assertFalse(w._flag_need_to_build_msg1)
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 1)
|
|
|
|
self.check_out(out[0], type=u"claim", nameplate=u"123")
|
|
|
|
|
|
|
|
# the server reveals the linked mailbox
|
|
|
|
response(w, type=u"claimed", mailbox=u"mb456")
|
|
|
|
|
|
|
|
# that triggers event_learned_mailbox, which should send open() and
|
|
|
|
# PAKE
|
2016-05-24 05:53:00 +00:00
|
|
|
self.assertEqual(w._mailbox_state, wormhole.OPEN)
|
2016-05-23 01:40:44 +00:00
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 2)
|
|
|
|
self.check_out(out[0], type=u"open", mailbox=u"mb456")
|
|
|
|
self.check_out(out[1], type=u"add", phase=u"pake")
|
|
|
|
self.assertNoResult(v)
|
|
|
|
|
|
|
|
# server echoes back all "add" messages
|
|
|
|
response(w, type=u"message", phase=u"pake", body=out[1][u"body"],
|
|
|
|
side=w._side)
|
|
|
|
self.assertNoResult(v)
|
|
|
|
|
2016-05-29 01:30:36 +00:00
|
|
|
# extract our outbound PAKE message
|
|
|
|
body = bytes_to_dict(hexstr_to_bytes(out[1][u"body"]))
|
|
|
|
msg1 = hexstr_to_bytes(body[u"pake_v1"])
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
# next we build the simulated peer's PAKE operation
|
|
|
|
side2 = w._side + u"other"
|
2016-05-29 01:30:36 +00:00
|
|
|
key, msg2 = self.make_pake(CODE, side2, msg1)
|
|
|
|
payload = {u"pake_v1": bytes_to_hexstr(msg2)}
|
|
|
|
body_hex = bytes_to_hexstr(dict_to_bytes(payload))
|
|
|
|
response(w, type=u"message", phase=u"pake", body=body_hex, side=side2)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
# hearing the peer's PAKE (msg2) makes us release the nameplate, send
|
2016-05-26 01:05:02 +00:00
|
|
|
# the confirmation message, and sends any queued phase messages. It
|
|
|
|
# doesn't deliver the verifier because we're still waiting on the
|
|
|
|
# confirmation message.
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertFalse(w._flag_need_to_see_mailbox_used)
|
|
|
|
self.assertEqual(w._key, key)
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 2, out)
|
|
|
|
self.check_out(out[0], type=u"release")
|
2016-05-26 02:09:00 +00:00
|
|
|
self.check_out(out[1], type=u"add", phase=u"version")
|
2016-05-26 01:05:02 +00:00
|
|
|
self.assertNoResult(v)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
# hearing a valid confirmation message doesn't throw an error
|
2016-05-26 01:27:37 +00:00
|
|
|
plaintext = json.dumps({}).encode("utf-8")
|
2016-05-26 02:09:00 +00:00
|
|
|
data_key = w._derive_phase_key(side2, u"version")
|
2016-05-26 01:27:37 +00:00
|
|
|
confmsg = w._encrypt_data(data_key, plaintext)
|
2016-05-26 02:13:37 +00:00
|
|
|
version2_hex = hexlify(confmsg).decode("ascii")
|
|
|
|
response(w, type=u"message", phase=u"version", body=version2_hex,
|
2016-05-23 01:40:44 +00:00
|
|
|
side=side2)
|
|
|
|
|
2016-05-26 01:05:02 +00:00
|
|
|
# and it releases the verifier
|
|
|
|
verifier = self.successResultOf(v)
|
|
|
|
self.assertEqual(verifier,
|
|
|
|
w.derive_key(u"wormhole:verifier", SecretBox.KEY_SIZE))
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
# an outbound message can now be sent immediately
|
|
|
|
w.send(b"phase0-outbound")
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 1)
|
|
|
|
self.check_out(out[0], type=u"add", phase=u"0")
|
|
|
|
# decrypt+check the outbound message
|
|
|
|
p0_outbound = unhexlify(out[0][u"body"].encode("ascii"))
|
2016-05-24 20:47:15 +00:00
|
|
|
msgkey0 = w._derive_phase_key(w._side, u"0")
|
2016-05-23 01:40:44 +00:00
|
|
|
p0_plaintext = w._decrypt_data(msgkey0, p0_outbound)
|
|
|
|
self.assertEqual(p0_plaintext, b"phase0-outbound")
|
|
|
|
|
|
|
|
# get() waits for the inbound message to arrive
|
|
|
|
md = w.get()
|
|
|
|
self.assertNoResult(md)
|
|
|
|
self.assertIn(u"0", w._receive_waiters)
|
|
|
|
self.assertNotIn(u"0", w._received_messages)
|
2016-05-24 20:47:15 +00:00
|
|
|
msgkey1 = w._derive_phase_key(side2, u"0")
|
|
|
|
p0_inbound = w._encrypt_data(msgkey1, b"phase0-inbound")
|
2016-05-23 01:40:44 +00:00
|
|
|
p0_inbound_hex = hexlify(p0_inbound).decode("ascii")
|
|
|
|
response(w, type=u"message", phase=u"0", body=p0_inbound_hex,
|
|
|
|
side=side2)
|
|
|
|
p0_in = self.successResultOf(md)
|
|
|
|
self.assertEqual(p0_in, b"phase0-inbound")
|
|
|
|
self.assertNotIn(u"0", w._receive_waiters)
|
|
|
|
self.assertIn(u"0", w._received_messages)
|
|
|
|
|
|
|
|
# receiving an inbound message will queue it until get() is called
|
2016-05-24 20:47:15 +00:00
|
|
|
msgkey2 = w._derive_phase_key(side2, u"1")
|
|
|
|
p1_inbound = w._encrypt_data(msgkey2, b"phase1-inbound")
|
2016-05-23 01:40:44 +00:00
|
|
|
p1_inbound_hex = hexlify(p1_inbound).decode("ascii")
|
|
|
|
response(w, type=u"message", phase=u"1", body=p1_inbound_hex,
|
|
|
|
side=side2)
|
|
|
|
self.assertIn(u"1", w._received_messages)
|
|
|
|
self.assertNotIn(u"1", w._receive_waiters)
|
|
|
|
p1_in = self.successResultOf(w.get())
|
|
|
|
self.assertEqual(p1_in, b"phase1-inbound")
|
|
|
|
self.assertIn(u"1", w._received_messages)
|
|
|
|
self.assertNotIn(u"1", w._receive_waiters)
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
|
|
|
self.assertNoResult(d)
|
2016-05-23 01:40:44 +00:00
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 1)
|
|
|
|
self.check_out(out[0], type=u"close", mood=u"happy")
|
2016-05-24 05:53:00 +00:00
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
|
|
|
|
response(w, type=u"released")
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
response(w, type=u"closed")
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
2016-05-24 05:53:00 +00:00
|
|
|
w._ws_closed(True, None, None)
|
2016-05-24 06:59:49 +00:00
|
|
|
self.assertEqual(self.successResultOf(d), None)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-23 07:14:39 +00:00
|
|
|
def test_close_wait_0(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
# Close before the connection is established. The connection still
|
|
|
|
# gets established, but it is then torn down before sending anything.
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
w._ws_closed(True, None, None)
|
|
|
|
self.successResultOf(d)
|
|
|
|
|
|
|
|
def test_close_wait_1(self):
|
2016-05-23 07:14:39 +00:00
|
|
|
# close before even claiming the nameplate
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
2016-05-23 07:14:39 +00:00
|
|
|
self.check_outbound(ws, [u"bind"])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
w._ws_closed(True, None, None)
|
|
|
|
self.successResultOf(d)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def test_close_wait_2(self):
|
|
|
|
# Close after claiming the nameplate, but before opening the mailbox.
|
|
|
|
# The 'claimed' response arrives before we close.
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
CODE = u"123-foo-bar"
|
|
|
|
w.set_code(CODE)
|
|
|
|
self.check_outbound(ws, [u"bind", u"claim"])
|
|
|
|
|
|
|
|
response(w, type=u"claimed", mailbox=u"mb123")
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
self.check_outbound(ws, [u"open", u"add", u"release", u"close"])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
|
|
|
|
response(w, type=u"released")
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
|
|
|
|
response(w, type=u"closed")
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
w._ws_closed(True, None, None)
|
|
|
|
self.successResultOf(d)
|
|
|
|
|
|
|
|
def test_close_wait_3(self):
|
2016-05-23 01:40:44 +00:00
|
|
|
# close after claiming the nameplate, but before opening the mailbox
|
2016-05-24 05:53:00 +00:00
|
|
|
# The 'claimed' response arrives after we start to close.
|
2016-05-23 01:40:44 +00:00
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
CODE = u"123-foo-bar"
|
|
|
|
w.set_code(CODE)
|
|
|
|
self.check_outbound(ws, [u"bind", u"claim"])
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
response(w, type=u"claimed", mailbox=u"mb123")
|
2016-05-23 01:40:44 +00:00
|
|
|
self.check_outbound(ws, [u"release"])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
|
|
|
|
response(w, type=u"released")
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
2016-05-23 01:45:50 +00:00
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
w._ws_closed(True, None, None)
|
|
|
|
self.successResultOf(d)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def test_close_wait_4(self):
|
2016-05-23 01:40:44 +00:00
|
|
|
# close after both claiming the nameplate and opening the mailbox
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
CODE = u"123-foo-bar"
|
|
|
|
w.set_code(CODE)
|
|
|
|
response(w, type=u"claimed", mailbox=u"mb456")
|
|
|
|
self.check_outbound(ws, [u"bind", u"claim", u"open", u"add"])
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
self.check_outbound(ws, [u"release", u"close"])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
|
|
|
|
response(w, type=u"released")
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
2016-05-23 01:45:50 +00:00
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
response(w, type=u"closed")
|
2016-05-23 01:45:50 +00:00
|
|
|
self.assertNoResult(d)
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
|
|
|
|
2016-05-23 01:45:50 +00:00
|
|
|
w._ws_closed(True, None, None)
|
|
|
|
self.successResultOf(d)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def test_close_wait_5(self):
|
2016-05-23 01:40:44 +00:00
|
|
|
# close after claiming the nameplate, opening the mailbox, then
|
|
|
|
# releasing the nameplate
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
CODE = u"123-foo-bar"
|
|
|
|
w.set_code(CODE)
|
|
|
|
response(w, type=u"claimed", mailbox=u"mb456")
|
|
|
|
|
|
|
|
w._key = b""
|
2016-05-24 20:47:15 +00:00
|
|
|
msgkey = w._derive_phase_key(u"side2", u"misc")
|
2016-05-23 01:40:44 +00:00
|
|
|
p1_inbound = w._encrypt_data(msgkey, b"")
|
|
|
|
p1_inbound_hex = hexlify(p1_inbound).decode("ascii")
|
|
|
|
response(w, type=u"message", phase=u"misc", side=u"side2",
|
|
|
|
body=p1_inbound_hex)
|
|
|
|
self.check_outbound(ws, [u"bind", u"claim", u"open", u"add",
|
|
|
|
u"release"])
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
d = w.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
self.check_outbound(ws, [u"close"])
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
|
|
|
|
|
|
|
response(w, type=u"released")
|
|
|
|
self.assertNoResult(d)
|
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [])
|
2016-05-23 01:45:50 +00:00
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
response(w, type=u"closed")
|
2016-05-23 01:45:50 +00:00
|
|
|
self.assertNoResult(d)
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(w._drop_connection.mock_calls, [mock.call()])
|
|
|
|
|
2016-05-23 01:45:50 +00:00
|
|
|
w._ws_closed(True, None, None)
|
|
|
|
self.successResultOf(d)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def test_close_errbacks(self):
|
|
|
|
# make sure the Deferreds returned by verify() and get() are properly
|
|
|
|
# errbacked upon close
|
|
|
|
pass
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
def test_get_code_mock(self):
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
ws = MockWebSocket() # TODO: mock w._ws_send_command instead
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
self.check_outbound(ws, [u"bind"])
|
|
|
|
|
|
|
|
gc_c = mock.Mock()
|
|
|
|
gc = gc_c.return_value = mock.Mock()
|
|
|
|
gc_d = gc.go.return_value = Deferred()
|
|
|
|
with mock.patch("wormhole.wormhole._GetCode", gc_c):
|
|
|
|
d = w.get_code()
|
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
gc_d.callback(u"123-foo-bar")
|
|
|
|
code = self.successResultOf(d)
|
|
|
|
self.assertEqual(code, u"123-foo-bar")
|
|
|
|
|
|
|
|
def test_get_code_real(self):
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
self.check_outbound(ws, [u"bind"])
|
|
|
|
|
|
|
|
d = w.get_code()
|
|
|
|
|
|
|
|
out = ws.outbound()
|
|
|
|
self.assertEqual(len(out), 1)
|
|
|
|
self.check_out(out[0], type=u"allocate")
|
|
|
|
# TODO: nameplate attributes go here
|
|
|
|
self.assertNoResult(d)
|
|
|
|
|
|
|
|
response(w, type=u"allocated", nameplate=u"123")
|
|
|
|
code = self.successResultOf(d)
|
|
|
|
self.assertIsInstance(code, type(u""))
|
|
|
|
self.assert_(code.startswith(u"123-"))
|
|
|
|
pieces = code.split(u"-")
|
|
|
|
self.assertEqual(len(pieces), 3) # nameplate plus two words
|
|
|
|
self.assert_(re.search(r'^\d+-\w+-\w+$', code), code)
|
|
|
|
|
2016-05-26 01:05:02 +00:00
|
|
|
# make sure verify() can be called both before and after the verifier is
|
|
|
|
# computed
|
|
|
|
|
|
|
|
def _test_verifier(self, when, order, success):
|
|
|
|
assert when in ("early", "middle", "late")
|
2016-05-26 02:13:37 +00:00
|
|
|
assert order in ("key-then-version", "version-then-key")
|
2016-05-26 01:05:02 +00:00
|
|
|
assert isinstance(success, bool)
|
|
|
|
#print(when, order, success)
|
|
|
|
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
w._ws_send_command = mock.Mock()
|
|
|
|
w._mailbox_state = wormhole.OPEN
|
2016-05-26 01:27:37 +00:00
|
|
|
side2 = u"side2"
|
2016-05-26 01:05:02 +00:00
|
|
|
d = None
|
|
|
|
|
|
|
|
if success:
|
|
|
|
w._key = b"key"
|
|
|
|
else:
|
|
|
|
w._key = b"wrongkey"
|
2016-05-26 01:27:37 +00:00
|
|
|
plaintext = json.dumps({}).encode("utf-8")
|
2016-05-26 02:09:00 +00:00
|
|
|
data_key = w._derive_phase_key(side2, u"version")
|
2016-05-26 01:27:37 +00:00
|
|
|
confmsg = w._encrypt_data(data_key, plaintext)
|
2016-05-26 01:05:02 +00:00
|
|
|
w._key = None
|
|
|
|
|
|
|
|
if when == "early":
|
|
|
|
d = w.verify()
|
|
|
|
self.assertNoResult(d)
|
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
if order == "key-then-version":
|
2016-05-26 01:05:02 +00:00
|
|
|
w._key = b"key"
|
|
|
|
w._event_established_key()
|
|
|
|
else:
|
2016-05-26 02:13:37 +00:00
|
|
|
w._event_received_version(side2, confmsg)
|
2016-05-26 01:05:02 +00:00
|
|
|
|
|
|
|
if when == "middle":
|
|
|
|
d = w.verify()
|
|
|
|
if d:
|
|
|
|
self.assertNoResult(d) # still waiting for other msg
|
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
if order == "version-then-key":
|
2016-05-26 01:05:02 +00:00
|
|
|
w._key = b"key"
|
|
|
|
w._event_established_key()
|
|
|
|
else:
|
2016-05-26 02:13:37 +00:00
|
|
|
w._event_received_version(side2, confmsg)
|
2016-05-26 01:05:02 +00:00
|
|
|
|
|
|
|
if when == "late":
|
|
|
|
d = w.verify()
|
|
|
|
if success:
|
|
|
|
self.successResultOf(d)
|
|
|
|
else:
|
|
|
|
self.assertFailure(d, wormhole.WrongPasswordError)
|
|
|
|
self.flushLoggedErrors(WrongPasswordError)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
def test_verifier(self):
|
2016-05-26 01:05:02 +00:00
|
|
|
for when in ("early", "middle", "late"):
|
2016-05-26 02:13:37 +00:00
|
|
|
for order in ("key-then-version", "version-then-key"):
|
2016-05-26 01:05:02 +00:00
|
|
|
for success in (False, True):
|
|
|
|
self._test_verifier(when, order, success)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
|
2016-05-23 07:14:39 +00:00
|
|
|
def test_api_errors(self):
|
|
|
|
# doing things you're not supposed to do
|
|
|
|
pass
|
|
|
|
|
|
|
|
def test_welcome_error(self):
|
|
|
|
# A welcome message could arrive at any time, with an [error] key
|
|
|
|
# that should make us halt. In practice, though, this gets sent as
|
|
|
|
# soon as the connection is established, which limits the possible
|
|
|
|
# states in which we might see it.
|
|
|
|
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
self.check_outbound(ws, [u"bind"])
|
|
|
|
|
|
|
|
d1 = w.get()
|
2016-05-24 05:53:00 +00:00
|
|
|
d2 = w.verify()
|
2016-05-23 07:14:39 +00:00
|
|
|
d3 = w.get_code()
|
|
|
|
# TODO (tricky): test w.input_code
|
|
|
|
|
|
|
|
self.assertNoResult(d1)
|
|
|
|
self.assertNoResult(d2)
|
|
|
|
self.assertNoResult(d3)
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
w._signal_error(WelcomeError(u"you are not actually welcome"), u"pouty")
|
|
|
|
self.failureResultOf(d1, WelcomeError)
|
|
|
|
self.failureResultOf(d2, WelcomeError)
|
|
|
|
self.failureResultOf(d3, WelcomeError)
|
2016-05-23 07:14:39 +00:00
|
|
|
|
|
|
|
# once the error is signalled, all API calls should fail
|
2016-05-24 05:53:00 +00:00
|
|
|
self.assertRaises(WelcomeError, w.send, u"foo")
|
|
|
|
self.assertRaises(WelcomeError,
|
|
|
|
w.derive_key, u"foo", SecretBox.KEY_SIZE)
|
|
|
|
self.failureResultOf(w.get(), WelcomeError)
|
|
|
|
self.failureResultOf(w.verify(), WelcomeError)
|
2016-05-23 07:14:39 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
def test_version_error(self):
|
2016-05-26 02:09:00 +00:00
|
|
|
# we should only receive the "version" message after we receive the
|
2016-05-23 07:14:39 +00:00
|
|
|
# PAKE message, by which point we should know the key. If the
|
|
|
|
# confirmation message doesn't decrypt, we signal an error.
|
|
|
|
timing = DebugTiming()
|
|
|
|
w = wormhole._Wormhole(APPID, u"relay_url", reactor, None, timing)
|
|
|
|
w._drop_connection = mock.Mock()
|
|
|
|
ws = MockWebSocket()
|
|
|
|
w._event_connected(ws)
|
|
|
|
w._event_ws_opened(None)
|
|
|
|
w.set_code(u"123-foo-bar")
|
|
|
|
response(w, type=u"claimed", mailbox=u"mb456")
|
|
|
|
|
|
|
|
d1 = w.get()
|
2016-05-24 05:53:00 +00:00
|
|
|
d2 = w.verify()
|
2016-05-23 07:14:39 +00:00
|
|
|
self.assertNoResult(d1)
|
|
|
|
self.assertNoResult(d2)
|
|
|
|
|
|
|
|
out = ws.outbound()
|
|
|
|
# [u"bind", u"claim", u"open", u"add"]
|
|
|
|
self.assertEqual(len(out), 4)
|
|
|
|
self.assertEqual(out[3][u"type"], u"add")
|
|
|
|
|
|
|
|
sp2 = SPAKE2_Symmetric(b"", idSymmetric=wormhole.to_bytes(APPID))
|
|
|
|
msg2 = sp2.start()
|
2016-05-29 01:30:36 +00:00
|
|
|
payload = {u"pake_v1": bytes_to_hexstr(msg2)}
|
|
|
|
body_hex = bytes_to_hexstr(dict_to_bytes(payload))
|
|
|
|
response(w, type=u"message", phase=u"pake", body=body_hex, side=u"s2")
|
2016-05-23 07:14:39 +00:00
|
|
|
self.assertNoResult(d1)
|
2016-05-26 01:05:02 +00:00
|
|
|
self.assertNoResult(d2) # verify() waits for confirmation
|
2016-05-23 07:14:39 +00:00
|
|
|
|
2016-05-26 02:13:37 +00:00
|
|
|
# sending a random version message will cause a confirmation error
|
2016-05-24 05:53:00 +00:00
|
|
|
confkey = w.derive_key(u"WRONG", SecretBox.KEY_SIZE)
|
2016-05-23 07:14:39 +00:00
|
|
|
nonce = os.urandom(wormhole.CONFMSG_NONCE_LENGTH)
|
2016-05-26 02:13:37 +00:00
|
|
|
badversion = wormhole.make_confmsg(confkey, nonce)
|
|
|
|
badversion_hex = hexlify(badversion).decode("ascii")
|
|
|
|
response(w, type=u"message", phase=u"version", body=badversion_hex,
|
2016-05-23 07:14:39 +00:00
|
|
|
side=u"s2")
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
self.failureResultOf(d1, WrongPasswordError)
|
2016-05-26 01:05:02 +00:00
|
|
|
self.failureResultOf(d2, WrongPasswordError)
|
2016-05-23 07:14:39 +00:00
|
|
|
|
|
|
|
# once the error is signalled, all API calls should fail
|
2016-05-24 05:53:00 +00:00
|
|
|
self.assertRaises(WrongPasswordError, w.send, u"foo")
|
|
|
|
self.assertRaises(WrongPasswordError,
|
|
|
|
w.derive_key, u"foo", SecretBox.KEY_SIZE)
|
|
|
|
self.failureResultOf(w.get(), WrongPasswordError)
|
|
|
|
self.failureResultOf(w.verify(), WrongPasswordError)
|
2016-05-23 07:14:39 +00:00
|
|
|
|
|
|
|
|
2016-05-23 01:40:44 +00:00
|
|
|
# event orderings to exercise:
|
|
|
|
#
|
|
|
|
# * normal sender: set_code, send_phase1, connected, claimed, learn_msg2,
|
|
|
|
# learn_phase1
|
|
|
|
# * normal receiver (argv[2]=code): set_code, connected, learn_msg1,
|
|
|
|
# learn_phase1, send_phase1,
|
|
|
|
# * normal receiver (readline): connected, input_code
|
|
|
|
# *
|
|
|
|
# * set_code, then connected
|
|
|
|
# * connected, receive_pake, send_phase, set_code
|
|
|
|
|
|
|
|
class Wormholes(ServerBase, unittest.TestCase):
|
|
|
|
# integration test, with a real server
|
|
|
|
|
|
|
|
def doBoth(self, d1, d2):
|
|
|
|
return gatherResults([d1, d2], True)
|
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_basic(self):
|
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code)
|
|
|
|
w1.send(b"data1")
|
|
|
|
w2.send(b"data2")
|
|
|
|
dataX = yield w1.get()
|
|
|
|
dataY = yield w2.get()
|
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_same_message(self):
|
|
|
|
# the two sides use random nonces for their messages, so it's ok for
|
|
|
|
# both to try and send the same body: they'll result in distinct
|
|
|
|
# encrypted messages
|
2016-05-24 05:53:00 +00:00
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
2016-05-23 01:40:44 +00:00
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code)
|
2016-05-24 05:53:00 +00:00
|
|
|
w1.send(b"data")
|
|
|
|
w2.send(b"data")
|
|
|
|
dataX = yield w1.get()
|
|
|
|
dataY = yield w2.get()
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(dataX, b"data")
|
|
|
|
self.assertEqual(dataY, b"data")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_interleaved(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
2016-05-23 01:40:44 +00:00
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code)
|
2016-05-24 05:53:00 +00:00
|
|
|
w1.send(b"data1")
|
|
|
|
dataY = yield w2.get()
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(dataY, b"data1")
|
2016-05-24 05:53:00 +00:00
|
|
|
d = w1.get()
|
|
|
|
w2.send(b"data2")
|
|
|
|
dataX = yield d
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(dataX, b"data2")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_unidirectional(self):
|
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code)
|
|
|
|
w1.send(b"data1")
|
|
|
|
dataY = yield w2.get()
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_early(self):
|
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w1.send(b"data1")
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
d = w2.get()
|
|
|
|
w1.set_code(u"123-abc-def")
|
|
|
|
w2.set_code(u"123-abc-def")
|
|
|
|
dataY = yield d
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_fixed_code(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
2016-05-23 01:40:44 +00:00
|
|
|
w1.set_code(u"123-purple-elephant")
|
|
|
|
w2.set_code(u"123-purple-elephant")
|
2016-05-24 05:53:00 +00:00
|
|
|
w1.send(b"data1"), w2.send(b"data2")
|
2016-05-23 01:40:44 +00:00
|
|
|
dl = yield self.doBoth(w1.get(), w2.get())
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_multiple_messages(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
2016-05-23 01:40:44 +00:00
|
|
|
w1.set_code(u"123-purple-elephant")
|
|
|
|
w2.set_code(u"123-purple-elephant")
|
2016-05-24 05:53:00 +00:00
|
|
|
w1.send(b"data1"), w2.send(b"data2")
|
|
|
|
w1.send(b"data3"), w2.send(b"data4")
|
2016-05-23 01:40:44 +00:00
|
|
|
dl = yield self.doBoth(w1.get(), w2.get())
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
|
|
|
dl = yield self.doBoth(w1.get(), w2.get())
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data4")
|
|
|
|
self.assertEqual(dataY, b"data3")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_wrong_password(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
2016-05-23 01:40:44 +00:00
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code+"not")
|
2016-05-24 05:53:00 +00:00
|
|
|
# That's enough to allow both sides to discover the mismatch, but
|
|
|
|
# only after the confirmation message gets through. API calls that
|
|
|
|
# don't wait will appear to work until the mismatched confirmation
|
|
|
|
# message arrives.
|
|
|
|
w1.send(b"should still work")
|
|
|
|
w2.send(b"should still work")
|
|
|
|
|
|
|
|
# API calls that wait (i.e. get) will errback
|
2016-05-23 01:40:44 +00:00
|
|
|
yield self.assertFailure(w2.get(), WrongPasswordError)
|
|
|
|
yield self.assertFailure(w1.get(), WrongPasswordError)
|
|
|
|
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
self.flushLoggedErrors(WrongPasswordError)
|
2016-05-23 01:40:44 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_verifier(self):
|
2016-05-24 05:53:00 +00:00
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
2016-05-23 01:40:44 +00:00
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code)
|
2016-05-24 05:53:00 +00:00
|
|
|
v1 = yield w1.verify()
|
|
|
|
v2 = yield w2.verify()
|
2016-05-23 01:40:44 +00:00
|
|
|
self.failUnlessEqual(type(v1), type(b""))
|
|
|
|
self.failUnlessEqual(v1, v2)
|
2016-05-24 05:53:00 +00:00
|
|
|
w1.send(b"data1")
|
|
|
|
w2.send(b"data2")
|
|
|
|
dataX = yield w1.get()
|
|
|
|
dataY = yield w2.get()
|
2016-05-23 01:40:44 +00:00
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|
2016-05-26 01:27:37 +00:00
|
|
|
@inlineCallbacks
|
|
|
|
def test_versions(self):
|
|
|
|
# there's no API for this yet, but make sure the internals work
|
|
|
|
w1 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w1._my_versions = {u"w1": 123}
|
|
|
|
w2 = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
w2._my_versions = {u"w2": 456}
|
|
|
|
code = yield w1.get_code()
|
|
|
|
w2.set_code(code)
|
|
|
|
yield w1.verify()
|
|
|
|
self.assertEqual(w1._their_versions, {u"w2": 456})
|
|
|
|
yield w2.verify()
|
|
|
|
self.assertEqual(w2._their_versions, {u"w1": 123})
|
|
|
|
yield w1.close()
|
|
|
|
yield w2.close()
|
|
|
|
|
2016-05-24 05:53:00 +00:00
|
|
|
class Errors(ServerBase, unittest.TestCase):
|
2016-05-23 01:40:44 +00:00
|
|
|
@inlineCallbacks
|
2016-05-24 05:53:00 +00:00
|
|
|
def test_codes_1(self):
|
|
|
|
w = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
# definitely too early
|
|
|
|
self.assertRaises(UsageError, w.derive_key, u"purpose", 12)
|
|
|
|
|
|
|
|
w.set_code(u"123-purple-elephant")
|
|
|
|
# code can only be set once
|
|
|
|
self.assertRaises(UsageError, w.set_code, u"123-nope")
|
|
|
|
yield self.assertFailure(w.get_code(), UsageError)
|
|
|
|
yield self.assertFailure(w.input_code(), UsageError)
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w.close()
|
2016-05-24 05:53:00 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def test_codes_2(self):
|
|
|
|
w = wormhole.wormhole(APPID, self.relayurl, reactor)
|
|
|
|
yield w.get_code()
|
|
|
|
self.assertRaises(UsageError, w.set_code, u"123-nope")
|
|
|
|
yield self.assertFailure(w.get_code(), UsageError)
|
|
|
|
yield self.assertFailure(w.input_code(), UsageError)
|
2016-05-24 06:59:49 +00:00
|
|
|
yield w.close()
|
2016-05-23 01:40:44 +00:00
|
|
|
|