-
Notifications
You must be signed in to change notification settings - Fork 223
/
Copy pathtrace2scen.py
97 lines (78 loc) · 2.81 KB
/
trace2scen.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
# Copyright (c) Microsoft Corporation. All rights reserved.
# Licensed under the Apache 2.0 License.
import sys
import json
import os
def comment(action):
return f"# {action['name']} {action['location']['module']}:{action['location']['beginLine']}"
def term(ctx, pre):
return str(pre["currentTerm"][ctx["i"]])
def noop(ctx, pre, post):
return ["# Noop"]
MAP = {
"ClientRequest": lambda ctx, pre, post: ["replicate", term(ctx, pre), "42"],
"MCClientRequest": lambda ctx, pre, post: ["replicate", term(ctx, pre), "42"],
"CheckQuorum": lambda ctx, pre, post: ["periodic_one", ctx["i"], "110"],
"Timeout": lambda ctx, pre, post: ["periodic_one", ctx["i"], "110"],
"MCTimeout": lambda ctx, pre, post: ["periodic_one", ctx["i"], "110"],
"RequestVote": noop,
"AppendEntries": lambda _, __, ___: ["dispatch_all"],
"BecomeLeader": noop,
"SignCommittableMessages": lambda ctx, pre, post: [
"emit_signature",
term(ctx, pre),
],
"MCSignCommittableMessages": lambda ctx, pre, post: [
"emit_signature",
term(ctx, pre),
],
"ChangeConfigurationInt": lambda ctx, pre, post: [
"replicate_new_configuration",
term(ctx, pre),
*ctx["newConfiguration"],
],
"AdvanceCommitIndex": noop,
"HandleRequestVoteRequest": lambda _, __, ___: ["dispatch_all"],
"HandleRequestVoteResponse": noop,
"RejectAppendEntriesRequest": noop,
"ReturnToFollowerState": noop,
"AppendEntriesAlreadyDone": noop,
"RcvDropIgnoredMessage": noop,
"RcvUpdateTerm": noop,
"RcvRequestVoteRequest": noop,
"RcvRequestVoteResponse": noop,
}
def post_commit(post):
return [
["assert_commit_idx", node, str(idx)]
for node, idx in post["commitIndex"].items()
]
def post_state(post):
entries = []
for node, state in post["state"].items():
entries.append(["assert_detail", "leadership_state", node, state])
return entries
def step_to_action(pre_state, action, post_state):
return os.linesep.join(
[
comment(action),
",".join(
MAP[action["name"]](action["context"], pre_state[1], post_state[1])
),
]
)
def asserts(pre_state, action, post_state, assert_gen):
return os.linesep.join(
[",".join(assertion) for assertion in assert_gen(post_state[1])]
)
if __name__ == "__main__":
with open(sys.argv[1]) as trace:
steps = json.load(trace)["action"]
initial_state = steps[0][0][1]
(initial_node,) = [node for node, log in initial_state["log"].items() if log]
print(f"start_node,{initial_node}")
print(f"emit_signature,2")
for step in steps:
print(step_to_action(*step))
print(asserts(*step, post_state))
print(asserts(*steps[-1], post_commit))