2015-10-07 00:20:12 +00:00
|
|
|
from __future__ import print_function
|
2015-09-28 07:37:35 +00:00
|
|
|
import sys, json
|
2015-06-21 02:18:21 +00:00
|
|
|
from twisted.trial import unittest
|
2015-10-07 00:20:12 +00:00
|
|
|
from twisted.internet.defer import gatherResults, succeed
|
2015-11-17 02:24:39 +00:00
|
|
|
from ..twisted.transcribe import (Wormhole, UsageError, ChannelManager,
|
|
|
|
WrongPasswordError)
|
2015-10-07 23:22:27 +00:00
|
|
|
from ..twisted.eventsource_twisted import EventSourceParser
|
2015-09-27 00:59:39 +00:00
|
|
|
from .common import ServerBase
|
2015-06-21 02:18:21 +00:00
|
|
|
|
2015-10-07 00:02:52 +00:00
|
|
|
APPID = u"appid"
|
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
class Channel(ServerBase, unittest.TestCase):
|
|
|
|
def ignore(self, welcome):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def test_allocate(self):
|
|
|
|
cm = ChannelManager(self.relayurl, APPID, u"side", self.ignore)
|
|
|
|
d = cm.list_channels()
|
|
|
|
def _got_channels(channels):
|
|
|
|
self.failUnlessEqual(channels, [])
|
|
|
|
d.addCallback(_got_channels)
|
|
|
|
d.addCallback(lambda _: cm.allocate())
|
|
|
|
def _allocated(channelid):
|
|
|
|
self.failUnlessEqual(type(channelid), int)
|
|
|
|
self._channelid = channelid
|
|
|
|
d.addCallback(_allocated)
|
|
|
|
d.addCallback(lambda _: cm.connect(self._channelid))
|
|
|
|
def _connected(c):
|
|
|
|
self._channel = c
|
|
|
|
d.addCallback(_connected)
|
2015-10-09 01:35:26 +00:00
|
|
|
d.addCallback(lambda _: self._channel.deallocate(u"happy"))
|
2015-10-07 00:20:12 +00:00
|
|
|
return d
|
|
|
|
|
|
|
|
def test_messages(self):
|
|
|
|
cm1 = ChannelManager(self.relayurl, APPID, u"side1", self.ignore)
|
|
|
|
cm2 = ChannelManager(self.relayurl, APPID, u"side2", self.ignore)
|
|
|
|
c1 = cm1.connect(1)
|
|
|
|
c2 = cm2.connect(1)
|
|
|
|
|
|
|
|
d = succeed(None)
|
|
|
|
d.addCallback(lambda _: c1.send(u"phase1", b"msg1"))
|
|
|
|
d.addCallback(lambda _: c2.get(u"phase1"))
|
|
|
|
d.addCallback(lambda msg: self.failUnlessEqual(msg, b"msg1"))
|
|
|
|
d.addCallback(lambda _: c2.send(u"phase1", b"msg2"))
|
|
|
|
d.addCallback(lambda _: c1.get(u"phase1"))
|
|
|
|
d.addCallback(lambda msg: self.failUnlessEqual(msg, b"msg2"))
|
|
|
|
# it's legal to fetch a phase multiple times, should be idempotent
|
|
|
|
d.addCallback(lambda _: c1.get(u"phase1"))
|
|
|
|
d.addCallback(lambda msg: self.failUnlessEqual(msg, b"msg2"))
|
|
|
|
# deallocating one side is not enough to destroy the channel
|
|
|
|
d.addCallback(lambda _: c2.deallocate())
|
|
|
|
def _not_yet(_):
|
|
|
|
self._relay_server.prune()
|
|
|
|
self.failUnlessEqual(len(self._relay_server._apps), 1)
|
|
|
|
d.addCallback(_not_yet)
|
|
|
|
# but deallocating both will make the messages go away
|
2015-10-09 01:35:26 +00:00
|
|
|
d.addCallback(lambda _: c1.deallocate(u"sad"))
|
2015-10-07 00:20:12 +00:00
|
|
|
def _gone(_):
|
|
|
|
self._relay_server.prune()
|
|
|
|
self.failUnlessEqual(len(self._relay_server._apps), 0)
|
|
|
|
d.addCallback(_gone)
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
2015-11-17 00:47:52 +00:00
|
|
|
def test_get_multiple_phases(self):
|
|
|
|
cm1 = ChannelManager(self.relayurl, APPID, u"side1", self.ignore)
|
|
|
|
cm2 = ChannelManager(self.relayurl, APPID, u"side2", self.ignore)
|
|
|
|
c1 = cm1.connect(1)
|
|
|
|
c2 = cm2.connect(1)
|
|
|
|
|
|
|
|
self.failUnlessRaises(TypeError, c2.get_first_of, u"phase1")
|
|
|
|
self.failUnlessRaises(TypeError, c2.get_first_of, [u"phase1", 7])
|
|
|
|
|
|
|
|
d = succeed(None)
|
|
|
|
d.addCallback(lambda _: c1.send(u"phase1", b"msg1"))
|
|
|
|
|
|
|
|
d.addCallback(lambda _: c2.get_first_of([u"phase1", u"phase2"]))
|
|
|
|
d.addCallback(lambda phase_and_body:
|
|
|
|
self.failUnlessEqual(phase_and_body,
|
|
|
|
(u"phase1", b"msg1")))
|
|
|
|
d.addCallback(lambda _: c2.get_first_of([u"phase2", u"phase1"]))
|
|
|
|
d.addCallback(lambda phase_and_body:
|
|
|
|
self.failUnlessEqual(phase_and_body,
|
|
|
|
(u"phase1", b"msg1")))
|
|
|
|
|
|
|
|
d.addCallback(lambda _: c1.send(u"phase2", b"msg2"))
|
|
|
|
d.addCallback(lambda _: c2.get(u"phase2"))
|
|
|
|
|
|
|
|
# if both are present, it should prefer the first one we asked for
|
|
|
|
d.addCallback(lambda _: c2.get_first_of([u"phase1", u"phase2"]))
|
|
|
|
d.addCallback(lambda phase_and_body:
|
|
|
|
self.failUnlessEqual(phase_and_body,
|
|
|
|
(u"phase1", b"msg1")))
|
|
|
|
d.addCallback(lambda _: c2.get_first_of([u"phase2", u"phase1"]))
|
|
|
|
d.addCallback(lambda phase_and_body:
|
|
|
|
self.failUnlessEqual(phase_and_body,
|
|
|
|
(u"phase2", b"msg2")))
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
2015-10-07 00:20:12 +00:00
|
|
|
def test_appid_independence(self):
|
|
|
|
APPID_A = u"appid_A"
|
|
|
|
APPID_B = u"appid_B"
|
|
|
|
cm1a = ChannelManager(self.relayurl, APPID_A, u"side1", self.ignore)
|
|
|
|
cm2a = ChannelManager(self.relayurl, APPID_A, u"side2", self.ignore)
|
|
|
|
c1a = cm1a.connect(1)
|
|
|
|
c2a = cm2a.connect(1)
|
|
|
|
cm1b = ChannelManager(self.relayurl, APPID_B, u"side1", self.ignore)
|
|
|
|
cm2b = ChannelManager(self.relayurl, APPID_B, u"side2", self.ignore)
|
|
|
|
c1b = cm1b.connect(1)
|
|
|
|
c2b = cm2b.connect(1)
|
|
|
|
|
|
|
|
d = succeed(None)
|
|
|
|
d.addCallback(lambda _: c1a.send(u"phase1", b"msg1a"))
|
|
|
|
d.addCallback(lambda _: c1b.send(u"phase1", b"msg1b"))
|
|
|
|
d.addCallback(lambda _: c2a.get(u"phase1"))
|
|
|
|
d.addCallback(lambda msg: self.failUnlessEqual(msg, b"msg1a"))
|
|
|
|
d.addCallback(lambda _: c2b.get(u"phase1"))
|
|
|
|
d.addCallback(lambda msg: self.failUnlessEqual(msg, b"msg1b"))
|
|
|
|
return d
|
|
|
|
|
2015-09-22 06:21:40 +00:00
|
|
|
class Basic(ServerBase, unittest.TestCase):
|
2015-10-04 05:03:27 +00:00
|
|
|
|
|
|
|
def doBoth(self, d1, d2):
|
|
|
|
return gatherResults([d1, d2], True)
|
|
|
|
|
2015-06-21 02:18:21 +00:00
|
|
|
def test_basic(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-06-21 02:18:21 +00:00
|
|
|
d = w1.get_code()
|
|
|
|
def _got_code(code):
|
|
|
|
w2.set_code(code)
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.send_data(b"data1"), w2.send_data(b"data2"))
|
2015-06-21 02:18:21 +00:00
|
|
|
d.addCallback(_got_code)
|
2015-10-04 05:03:27 +00:00
|
|
|
def _sent(res):
|
|
|
|
return self.doBoth(w1.get_data(), w2.get_data())
|
|
|
|
d.addCallback(_sent)
|
2015-06-21 02:18:21 +00:00
|
|
|
def _done(dl):
|
2015-09-28 23:44:00 +00:00
|
|
|
(dataX, dataY) = dl
|
2015-09-28 06:09:51 +00:00
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.close(), w2.close())
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2015-10-07 23:28:11 +00:00
|
|
|
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
|
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
|
|
|
d = w1.get_code()
|
|
|
|
def _got_code(code):
|
|
|
|
w2.set_code(code)
|
|
|
|
return self.doBoth(w1.send_data(b"data"), w2.send_data(b"data"))
|
|
|
|
d.addCallback(_got_code)
|
|
|
|
def _sent(res):
|
|
|
|
return self.doBoth(w1.get_data(), w2.get_data())
|
|
|
|
d.addCallback(_sent)
|
|
|
|
def _done(dl):
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data")
|
|
|
|
self.assertEqual(dataY, b"data")
|
|
|
|
return self.doBoth(w1.close(), w2.close())
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2015-10-04 05:03:27 +00:00
|
|
|
def test_interleaved(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-10-04 05:03:27 +00:00
|
|
|
d = w1.get_code()
|
|
|
|
def _got_code(code):
|
|
|
|
w2.set_code(code)
|
|
|
|
return self.doBoth(w1.send_data(b"data1"), w2.get_data())
|
|
|
|
d.addCallback(_got_code)
|
|
|
|
def _sent(res):
|
|
|
|
(_, dataY) = res
|
|
|
|
self.assertEqual(dataY, b"data1")
|
|
|
|
return self.doBoth(w1.get_data(), w2.send_data(b"data2"))
|
|
|
|
d.addCallback(_sent)
|
|
|
|
def _done(dl):
|
|
|
|
(dataX, _) = dl
|
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
return self.doBoth(w1.close(), w2.close())
|
2015-06-21 02:18:21 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_fixed_code(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-10-07 02:42:10 +00:00
|
|
|
w1.set_code(u"123-purple-elephant")
|
|
|
|
w2.set_code(u"123-purple-elephant")
|
2015-10-04 05:03:27 +00:00
|
|
|
d = self.doBoth(w1.send_data(b"data1"), w2.send_data(b"data2"))
|
|
|
|
def _sent(res):
|
|
|
|
return self.doBoth(w1.get_data(), w2.get_data())
|
|
|
|
d.addCallback(_sent)
|
2015-06-21 02:18:21 +00:00
|
|
|
def _done(dl):
|
2015-09-28 23:44:00 +00:00
|
|
|
(dataX, dataY) = dl
|
2015-09-28 06:09:51 +00:00
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.close(), w2.close())
|
2015-06-21 02:18:21 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2015-10-07 03:39:20 +00:00
|
|
|
def test_phases(self):
|
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
|
|
|
w1.set_code(u"123-purple-elephant")
|
|
|
|
w2.set_code(u"123-purple-elephant")
|
|
|
|
d = self.doBoth(w1.send_data(b"data1", u"p1"),
|
|
|
|
w2.send_data(b"data2", u"p1"))
|
|
|
|
d.addCallback(lambda _:
|
|
|
|
self.doBoth(w1.send_data(b"data3", u"p2"),
|
|
|
|
w2.send_data(b"data4", u"p2")))
|
|
|
|
d.addCallback(lambda _:
|
|
|
|
self.doBoth(w1.get_data(u"p2"),
|
|
|
|
w2.get_data(u"p1")))
|
|
|
|
def _got_1(dl):
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data4")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
|
|
|
return self.doBoth(w1.get_data(u"p1"),
|
|
|
|
w2.get_data(u"p2"))
|
|
|
|
d.addCallback(_got_1)
|
|
|
|
def _got_2(dl):
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data3")
|
|
|
|
return self.doBoth(w1.close(), w2.close())
|
|
|
|
d.addCallback(_got_2)
|
|
|
|
return d
|
|
|
|
|
2015-11-17 02:24:39 +00:00
|
|
|
def test_wrong_password(self):
|
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
|
|
|
d = w1.get_code()
|
|
|
|
d.addCallback(lambda code: w2.set_code(code+"not"))
|
|
|
|
|
|
|
|
# w2 can't throw WrongPasswordError until it sees a CONFIRM message,
|
|
|
|
# and w1 won't send CONFIRM until it sees a PAKE message, which w2
|
|
|
|
# won't send until we call get_data. So we need both sides to be
|
|
|
|
# running at the same time for this test.
|
|
|
|
def _w1_sends():
|
|
|
|
return w1.send_data(b"data1")
|
|
|
|
def _w2_gets():
|
|
|
|
return self.assertFailure(w2.get_data(), WrongPasswordError)
|
|
|
|
d.addCallback(lambda _: self.doBoth(_w1_sends(), _w2_gets()))
|
|
|
|
|
|
|
|
# and now w1 should have enough information to throw too
|
|
|
|
d.addCallback(lambda _: self.assertFailure(w1.get_data(),
|
|
|
|
WrongPasswordError))
|
|
|
|
def _done(_):
|
|
|
|
# both sides are closed automatically upon error, but it's still
|
|
|
|
# legal to call .close(), and should be idempotent
|
|
|
|
return self.doBoth(w1.close(), w2.close())
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
2015-09-28 23:49:36 +00:00
|
|
|
def test_verifier(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-09-28 23:49:36 +00:00
|
|
|
d = w1.get_code()
|
|
|
|
def _got_code(code):
|
|
|
|
w2.set_code(code)
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.get_verifier(), w2.get_verifier())
|
2015-09-28 23:49:36 +00:00
|
|
|
d.addCallback(_got_code)
|
|
|
|
def _check_verifier(res):
|
|
|
|
v1, v2 = res
|
|
|
|
self.failUnlessEqual(type(v1), type(b""))
|
|
|
|
self.failUnlessEqual(v1, v2)
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.send_data(b"data1"), w2.send_data(b"data2"))
|
2015-09-28 23:49:36 +00:00
|
|
|
d.addCallback(_check_verifier)
|
2015-10-04 05:03:27 +00:00
|
|
|
def _sent(res):
|
|
|
|
return self.doBoth(w1.get_data(), w2.get_data())
|
|
|
|
d.addCallback(_sent)
|
2015-09-28 23:49:36 +00:00
|
|
|
def _done(dl):
|
|
|
|
(dataX, dataY) = dl
|
|
|
|
self.assertEqual(dataX, b"data2")
|
|
|
|
self.assertEqual(dataY, b"data1")
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.close(), w2.close())
|
2015-09-28 23:49:36 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def test_verifier_mismatch(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-09-28 23:49:36 +00:00
|
|
|
d = w1.get_code()
|
|
|
|
def _got_code(code):
|
|
|
|
w2.set_code(code+"not")
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.get_verifier(), w2.get_verifier())
|
2015-09-28 23:49:36 +00:00
|
|
|
d.addCallback(_got_code)
|
|
|
|
def _check_verifier(res):
|
|
|
|
v1, v2 = res
|
|
|
|
self.failUnlessEqual(type(v1), type(b""))
|
|
|
|
self.failIfEqual(v1, v2)
|
2015-10-04 05:03:27 +00:00
|
|
|
return self.doBoth(w1.close(), w2.close())
|
2015-09-28 23:49:36 +00:00
|
|
|
d.addCallback(_check_verifier)
|
|
|
|
return d
|
|
|
|
|
2015-06-21 02:18:21 +00:00
|
|
|
def test_errors(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
2015-06-21 02:18:21 +00:00
|
|
|
self.assertRaises(UsageError, w1.get_verifier)
|
2015-10-04 05:03:27 +00:00
|
|
|
self.assertRaises(UsageError, w1.send_data, b"data")
|
|
|
|
self.assertRaises(UsageError, w1.get_data)
|
2015-10-07 02:42:10 +00:00
|
|
|
w1.set_code(u"123-purple-elephant")
|
2015-10-07 03:52:47 +00:00
|
|
|
self.assertRaises(UsageError, w1.set_code, u"123-nope")
|
2015-06-21 02:18:21 +00:00
|
|
|
self.assertRaises(UsageError, w1.get_code)
|
2015-10-07 00:02:52 +00:00
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-06-21 02:18:21 +00:00
|
|
|
d = w2.get_code()
|
|
|
|
self.assertRaises(UsageError, w2.get_code)
|
2015-10-04 05:03:27 +00:00
|
|
|
def _got_code(code):
|
|
|
|
return self.doBoth(w1.close(), w2.close())
|
|
|
|
d.addCallback(_got_code)
|
2015-06-21 02:18:21 +00:00
|
|
|
return d
|
|
|
|
|
2015-10-07 03:39:20 +00:00
|
|
|
def test_repeat_phases(self):
|
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
|
|
|
w1.set_code(u"123-purple-elephant")
|
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
|
|
|
w2.set_code(u"123-purple-elephant")
|
|
|
|
# we must let them establish a key before we can send data
|
|
|
|
d = self.doBoth(w1.get_verifier(), w2.get_verifier())
|
|
|
|
d.addCallback(lambda _: w1.send_data(b"data1", phase=u"1"))
|
|
|
|
def _sent(res):
|
2015-11-17 00:20:00 +00:00
|
|
|
# underscore-prefixed phases are reserved
|
|
|
|
self.assertRaises(UsageError, w1.send_data, b"data1", phase=u"_1")
|
|
|
|
self.assertRaises(UsageError, w1.get_data, phase=u"_1")
|
2015-10-07 03:39:20 +00:00
|
|
|
# you can't send twice to the same phase
|
|
|
|
self.assertRaises(UsageError, w1.send_data, b"data1", phase=u"1")
|
|
|
|
# but you can send to a different one
|
|
|
|
return w1.send_data(b"data2", phase=u"2")
|
|
|
|
d.addCallback(_sent)
|
|
|
|
d.addCallback(lambda _: w2.get_data(phase=u"1"))
|
|
|
|
def _got1(res):
|
|
|
|
self.failUnlessEqual(res, b"data1")
|
|
|
|
# and you can't read twice from the same phase
|
|
|
|
self.assertRaises(UsageError, w2.get_data, phase=u"1")
|
|
|
|
# but you can read from a different one
|
|
|
|
return w2.get_data(phase=u"2")
|
|
|
|
d.addCallback(_got1)
|
|
|
|
def _got2(res):
|
|
|
|
self.failUnlessEqual(res, b"data2")
|
|
|
|
return self.doBoth(w1.close(), w2.close())
|
|
|
|
d.addCallback(_got2)
|
|
|
|
return d
|
|
|
|
|
2015-06-21 02:18:21 +00:00
|
|
|
def test_serialize(self):
|
2015-10-07 00:02:52 +00:00
|
|
|
w1 = Wormhole(APPID, self.relayurl)
|
2015-06-21 02:18:21 +00:00
|
|
|
self.assertRaises(UsageError, w1.serialize) # too early
|
2015-10-07 00:02:52 +00:00
|
|
|
w2 = Wormhole(APPID, self.relayurl)
|
2015-06-21 02:18:21 +00:00
|
|
|
d = w1.get_code()
|
|
|
|
def _got_code(code):
|
|
|
|
self.assertRaises(UsageError, w2.serialize) # too early
|
|
|
|
w2.set_code(code)
|
|
|
|
w2.serialize() # ok
|
|
|
|
s = w1.serialize()
|
|
|
|
self.assertEqual(type(s), type(""))
|
|
|
|
unpacked = json.loads(s) # this is supposed to be JSON
|
|
|
|
self.assertEqual(type(unpacked), dict)
|
2015-10-04 05:03:27 +00:00
|
|
|
self.new_w1 = Wormhole.from_serialized(s)
|
|
|
|
return self.doBoth(self.new_w1.send_data(b"data1"),
|
|
|
|
w2.send_data(b"data2"))
|
2015-06-21 02:18:21 +00:00
|
|
|
d.addCallback(_got_code)
|
2015-10-04 05:03:27 +00:00
|
|
|
def _sent(res):
|
|
|
|
return self.doBoth(self.new_w1.get_data(), w2.get_data())
|
|
|
|
d.addCallback(_sent)
|
2015-06-21 02:18:21 +00:00
|
|
|
def _done(dl):
|
2015-09-28 23:44:00 +00:00
|
|
|
(dataX, dataY) = dl
|
2015-10-04 05:03:27 +00:00
|
|
|
self.assertEqual((dataX, dataY), (b"data2", b"data1"))
|
2015-06-21 02:18:21 +00:00
|
|
|
self.assertRaises(UsageError, w2.serialize) # too late
|
2015-10-04 05:03:27 +00:00
|
|
|
return gatherResults([w1.close(), w2.close(), self.new_w1.close()],
|
|
|
|
True)
|
2015-06-21 02:18:21 +00:00
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
2015-09-28 07:37:35 +00:00
|
|
|
|
2015-10-07 23:22:27 +00:00
|
|
|
data1 = u"""\
|
|
|
|
event: welcome
|
|
|
|
data: one and a
|
|
|
|
data: two
|
|
|
|
data:.
|
|
|
|
|
|
|
|
data: three
|
|
|
|
|
|
|
|
: this line is ignored
|
|
|
|
event: e2
|
|
|
|
: this line is ignored too
|
|
|
|
i am a dataless field name
|
|
|
|
data: four
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
class FakeTransport:
|
|
|
|
disconnecting = False
|
|
|
|
|
|
|
|
class EventSourceClient(unittest.TestCase):
|
|
|
|
def test_parser(self):
|
|
|
|
events = []
|
|
|
|
p = EventSourceParser(lambda t,d: events.append((t,d)))
|
|
|
|
p.transport = FakeTransport()
|
|
|
|
p.dataReceived(data1)
|
|
|
|
self.failUnlessEqual(events,
|
|
|
|
[(u"welcome", u"one and a\ntwo\n."),
|
|
|
|
(u"message", u"three"),
|
|
|
|
(u"e2", u"four"),
|
|
|
|
])
|
|
|
|
|
2015-09-28 07:37:35 +00:00
|
|
|
if sys.version_info[0] >= 3:
|
2015-10-07 00:20:12 +00:00
|
|
|
Channel.skip = "twisted is not yet sufficiently ported to py3"
|
2015-09-28 07:37:35 +00:00
|
|
|
Basic.skip = "twisted is not yet sufficiently ported to py3"
|
2015-10-07 23:22:27 +00:00
|
|
|
EventSourceClient.skip = "twisted is not yet sufficiently ported to py3"
|
2015-09-28 07:37:35 +00:00
|
|
|
# as of 15.4.0, Twisted is still missing:
|
|
|
|
# * web.client.Agent (for all non-EventSource POSTs in transcribe.py)
|
|
|
|
# * python.logfile (to allow daemonization of 'wormhole server')
|