rewrite debug tracing, add to all machines

This commit is contained in:
Brian Warner 2017-03-01 19:55:13 -08:00
parent b0c9c9bb4c
commit 4793208d4e
13 changed files with 63 additions and 18 deletions

View File

@ -34,6 +34,8 @@ class Boss(object):
_journal = attrib(validator=provides(_interfaces.IJournal)) _journal = attrib(validator=provides(_interfaces.IJournal))
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._N = Nameplate() self._N = Nameplate()
@ -70,6 +72,17 @@ class Boss(object):
def start(self): def start(self):
self._RC.start() self._RC.start()
def _set_trace(self, client_name, which, logger):
names = {"B": self, "N": self._N, "M": self._M, "S": self._S,
"O": self._O, "K": self._K, "R": self._R,
"RC": self._RC, "NL": self._NL, "C": self._C,
"T": self._T}
for machine in which.split():
def tracer(old_state, input, new_state, machine=machine):
print("%s.%s[%s].%s -> [%s]" % (client_name, machine,
old_state, input, new_state))
names[machine].set_trace(tracer)
# and these are the state-machine transition functions, which don't take # and these are the state-machine transition functions, which don't take
# args # args
@m.state(initial=True) @m.state(initial=True)

View File

@ -25,6 +25,8 @@ def make_code(nameplate, code_length):
class Code(object): class Code(object):
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def wire(self, boss, rendezvous_connector, nameplate_lister): def wire(self, boss, rendezvous_connector, nameplate_lister):
self._B = _interfaces.IBoss(boss) self._B = _interfaces.IBoss(boss)

View File

@ -59,6 +59,8 @@ class Key(object):
_side = attrib(validator=instance_of(type(u""))) _side = attrib(validator=instance_of(type(u"")))
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def wire(self, boss, mailbox, receive): def wire(self, boss, mailbox, receive):
self._B = _interfaces.IBoss(boss) self._B = _interfaces.IBoss(boss)

View File

@ -11,7 +11,7 @@ class Mailbox(object):
_side = attrib(validator=instance_of(type(u""))) _side = attrib(validator=instance_of(type(u"")))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace() @m.setTrace()
def setTrace(): pass def set_trace(): pass
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._mailbox = None self._mailbox = None

View File

@ -7,7 +7,7 @@ from . import _interfaces
class Nameplate(object): class Nameplate(object):
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace() @m.setTrace()
def setTrace(): pass def set_trace(): pass
def __init__(self): def __init__(self):
self._nameplate = None self._nameplate = None

View File

@ -6,6 +6,8 @@ from . import _interfaces
@implementer(_interfaces.INameplateLister) @implementer(_interfaces.INameplateLister)
class NameplateListing(object): class NameplateListing(object):
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def wire(self, rendezvous_connector, code): def wire(self, rendezvous_connector, code):
self._RC = _interfaces.IRendezvousConnector(rendezvous_connector) self._RC = _interfaces.IRendezvousConnector(rendezvous_connector)

View File

@ -11,6 +11,8 @@ class Order(object):
_side = attrib(validator=instance_of(type(u""))) _side = attrib(validator=instance_of(type(u"")))
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._key = None self._key = None

View File

@ -12,6 +12,8 @@ class Receive(object):
_side = attrib(validator=instance_of(type(u""))) _side = attrib(validator=instance_of(type(u"")))
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._key = None self._key = None

View File

@ -70,9 +70,9 @@ class RendezvousConnector(object):
_reactor = attrib() _reactor = attrib()
_journal = attrib(validator=provides(_interfaces.IJournal)) _journal = attrib(validator=provides(_interfaces.IJournal))
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
DEBUG = True
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._trace = None
self._ws = None self._ws = None
f = WSFactory(self, self._url) f = WSFactory(self, self._url)
f.setProtocolOptions(autoPingInterval=60, autoPingTimeout=600) f.setProtocolOptions(autoPingInterval=60, autoPingTimeout=600)
@ -80,6 +80,12 @@ class RendezvousConnector(object):
ep = self._make_endpoint(p.hostname, p.port or 80) ep = self._make_endpoint(p.hostname, p.port or 80)
self._connector = internet.ClientService(ep, f) self._connector = internet.ClientService(ep, f)
def set_trace(self, f):
self._trace = f
def _debug(self, what):
if self._trace:
self._trace(old_state="", input=what, new_state="")
def _make_endpoint(self, hostname, port): def _make_endpoint(self, hostname, port):
# TODO: Tor goes here # TODO: Tor goes here
return endpoints.HostnameEndpoint(self._reactor, hostname, port) return endpoints.HostnameEndpoint(self._reactor, hostname, port)
@ -130,8 +136,7 @@ class RendezvousConnector(object):
# from our WSClient (the WebSocket protocol) # from our WSClient (the WebSocket protocol)
def ws_open(self, proto): def ws_open(self, proto):
if self.DEBUG: self._debug("R.connected")
dmsg(self._side, "R.connected")
self._ws = proto self._ws = proto
self._tx("bind", appid=self._appid, side=self._side) self._tx("bind", appid=self._appid, side=self._side)
self._C.connected() self._C.connected()
@ -141,11 +146,17 @@ class RendezvousConnector(object):
def ws_message(self, payload): def ws_message(self, payload):
msg = bytes_to_dict(payload) msg = bytes_to_dict(payload)
if self.DEBUG and msg["type"]!="ack": #if self.DEBUG and msg["type"]!="ack":
dmsg(self._side, "R.rx(%s %s%s)" % # dmsg(self._side, "R.rx(%s %s%s)" %
# (msg["type"], msg.get("phase",""),
# "[mine]" if msg.get("side","") == self._side else "",
# ))
if msg["type"] != "ack":
self._debug("R.rx(%s %s%s)" %
(msg["type"], msg.get("phase",""), (msg["type"], msg.get("phase",""),
"[mine]" if msg.get("side","") == self._side else "", "[mine]" if msg.get("side","") == self._side else "",
)) ))
self._timing.add("ws_receive", _side=self._side, message=msg) self._timing.add("ws_receive", _side=self._side, message=msg)
mtype = msg["type"] mtype = msg["type"]
meth = getattr(self, "_response_handle_"+mtype, None) meth = getattr(self, "_response_handle_"+mtype, None)
@ -156,8 +167,7 @@ class RendezvousConnector(object):
return meth(msg) return meth(msg)
def ws_close(self, wasClean, code, reason): def ws_close(self, wasClean, code, reason):
if self.DEBUG: self._debug("R.lost")
dmsg(self._side, "R.lost")
self._ws = None self._ws = None
self._C.lost() self._C.lost()
self._N.lost() self._N.lost()
@ -176,8 +186,7 @@ class RendezvousConnector(object):
# are so few messages, 16 bits is enough to be mostly-unique. # are so few messages, 16 bits is enough to be mostly-unique.
kwargs["id"] = bytes_to_hexstr(os.urandom(2)) kwargs["id"] = bytes_to_hexstr(os.urandom(2))
kwargs["type"] = mtype kwargs["type"] = mtype
if self.DEBUG: self._debug("R.tx(%s %s)" % (mtype.upper(), kwargs.get("phase", "")))
dmsg(self._side, "R.tx(%s %s)" % (mtype.upper(), kwargs.get("phase", "")))
payload = dict_to_bytes(kwargs) payload = dict_to_bytes(kwargs)
self._timing.add("ws_send", _side=self._side, **kwargs) self._timing.add("ws_send", _side=self._side, **kwargs)
self._ws.sendMessage(payload, False) self._ws.sendMessage(payload, False)

View File

@ -12,6 +12,8 @@ class Send(object):
_side = attrib(validator=instance_of(type(u""))) _side = attrib(validator=instance_of(type(u"")))
_timing = attrib(validator=provides(_interfaces.ITiming)) _timing = attrib(validator=provides(_interfaces.ITiming))
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace()
def set_trace(): pass
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._queue = [] self._queue = []

View File

@ -7,7 +7,7 @@ from . import _interfaces
class Terminator(object): class Terminator(object):
m = MethodicalMachine() m = MethodicalMachine()
@m.setTrace() @m.setTrace()
def setTrace(): pass def set_trace(): pass
def __attrs_post_init__(self): def __attrs_post_init__(self):
self._mood = None self._mood = None

View File

@ -26,6 +26,7 @@ class New(ServerBase, unittest.TestCase):
@inlineCallbacks @inlineCallbacks
def test_allocate(self): def test_allocate(self):
w = wormhole.deferred_wormhole(APPID, self.relayurl, reactor) w = wormhole.deferred_wormhole(APPID, self.relayurl, reactor)
w.debug_set_trace("W1")
w.allocate_code(2) w.allocate_code(2)
code = yield w.when_code() code = yield w.when_code()
print("code:", code) print("code:", code)
@ -40,9 +41,7 @@ class New(ServerBase, unittest.TestCase):
@inlineCallbacks @inlineCallbacks
def test_basic(self): def test_basic(self):
w1 = wormhole.deferred_wormhole(APPID, self.relayurl, reactor) w1 = wormhole.deferred_wormhole(APPID, self.relayurl, reactor)
def trace(old_state, input, new_state): w1.debug_set_trace("W1")
print("W1._M[%s].%s -> [%s]" % (old_state, input, new_state))
w1._boss._M.setTrace(trace)
w1.allocate_code(2) w1.allocate_code(2)
code = yield w1.when_code() code = yield w1.when_code()
print("code:", code) print("code:", code)

View File

@ -29,6 +29,10 @@ from ._boss import Boss, WormholeError
# wormhole(delegate=app, delegate_prefix="wormhole_", # wormhole(delegate=app, delegate_prefix="wormhole_",
# delegate_args=(args, kwargs)) # delegate_args=(args, kwargs))
def _log(client_name, machine_name, old_state, input, new_state):
print("%s.%s[%s].%s -> [%s]" % (client_name, machine_name,
old_state, input, new_state))
@attrs @attrs
@implementer(IWormhole) @implementer(IWormhole)
class _DelegatedWormhole(object): class _DelegatedWormhole(object):
@ -51,6 +55,10 @@ class _DelegatedWormhole(object):
def close(self): def close(self):
self._boss.close() self._boss.close()
def debug_set_trace(self, client_name, which="B N M S O K R RC NL C T",
logger=_log):
self._boss.set_trace(client_name, which, logger)
# from below # from below
def got_code(self, code): def got_code(self, code):
self._delegate.wormhole_got_code(code) self._delegate.wormhole_got_code(code)
@ -115,6 +123,10 @@ class _DeferredWormhole(object):
self._closed_observers.append(d) self._closed_observers.append(d)
return d return d
def debug_set_trace(self, client_name, which="B N M S O K R RC NL C T",
logger=_log):
self._boss._set_trace(client_name, which, logger)
# from below # from below
def got_code(self, code): def got_code(self, code):
self._code = code self._code = code