digraph { /* could shave a RTT by committing to the nameplate early, before finishing the rest of the code input. While the user is still typing/completing the code, we claim the nameplate, open the mailbox, and retrieve the peer's PAKE message. Then as soon as the user finishes entering the code, we build our own PAKE message, send PAKE, compute the key, send VERSION. Starting from the Return, this saves two round trips. OTOH it adds consequences to hitting Tab. */ start [label="Boss\n(manager)" style="dotted"] {rank=same; P0_code S0} P0_code [shape="box" style="dashed" label="input -> Code.input\n or allocate -> Code.allocate\n or set_code -> Code.set_code"] P0_code -> S0 S0 [label="S0: empty"] S0 -> P0_build [label="set_code"] S0 -> P_close_lonely [label="close"] P0_build [shape="box" label="W.got_code\nM.set_nameplate\nK.got_code"] P0_build -> S1 S1 [label="S1: lonely" color="orange"] S1 -> S2 [label="happy"] S1 -> P_close_scary [label="scared" color="red"] S1 -> P_close_lonely [label="close"] P_close_lonely [shape="box" label="M.close(lonely)"] P_close_lonely -> S_closing P_close_scary [shape="box" label="M.close(scary)" color="red"] P_close_scary -> S_closing [color="red"] S2 [label="S2: happy" color="green"] S2 -> P2_close [label="close"] P2_close [shape="box" label="M.close(happy)"] P2_close -> S_closing S2 -> P2_got_message [label="got_message"] P2_got_message [shape="box" label="A.received"] P2_got_message -> S2 S2 -> P_close_scary [label="scared" color="red"] S_closing [label="closing"] S_closing -> P_closed [label="closed"] S_closing -> S_closing [label="got_message\nhappy\nscared\nclose"] P_closed [shape="box" label="A.closed(reason)"] P_closed -> S_closed S_closed [label="closed"] {rank=same; Other S_closed} Other [shape="box" style="dashed" label="rx_welcome -> process\nsend -> S.send\ngot_verifier -> A.got_verifier\nallocate -> C.allocate\ninput -> C.input\nset_code -> C.set_code" ] }