-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathproto.awk
89 lines (79 loc) · 2.22 KB
/
proto.awk
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
BEGIN {
a = 0
tsi = 0
ar="[]"
tr[ar"byte"]="bytes"
tr["abstract.Point"]="bytes"
tr["StateAction"]="int"
tr["SubID"]="bytes"
tr["Nonce"]="bytes"
tr["Version"]="sint32"
print "syntax = \"proto2\";"
}
# a is the state
# state 0: look for proto start
a == 0 && /PROTOSTART/{ a = 1; next }
# state 1: in a protostart block
# store types for later replacements
a == 1 && /^\/\/ type/ { split($0, t, /:/); tsi++; ts[tsi, 1] = t[2]; ts[tsi, 2] = t[3]; next}
# copy all other comment lines at the beginning
a == 1 && /^\/\// { sub( "^// *", "" ); print; next }
# go on if non-comment line
a == 1 { a = 2; print; next }
# state 2: looking for a type structure, start it.
a == 2 && /^type.*struct/ { print "message", toupper(substr($2,1,1)) substr($2,2), "{"; a = 3; i = 1; next }
a == 2 { print; next }
# state 3: processing fields of the struct
# detect end of struct -> state 2
a == 3 && /^\}/ { print; a = 2; next }
# detect "// optional" tag in Go -> state 4
a == 3 && / *\/\/ optional/ { a = 4; next }
# ignore hidden fields
a == 3 && /^[[:blank:]]*[[:lower:]]/ { next }
# copy comments through
a == 3 && /[[:blank:]]*\/\// { sub( "[[:blank:]]*//[[:blank:]]*", "" ); print " //", $0; next }
a == 3 && /\*/ { sub( "\\*", "", $2 )
print_field("optional", $2, $1, i)
i = i + 1
next
}
a == 3 {
print_field("required", $2, $1, i)
i = i + 1
next
}
# state 4: manual optional
a == 4 { sub(/\/\/.*/, "", $2)
print_field("optional", $2, $1, i)
i = i + 1
a = 3
next
}
function print_field( optional, typ, name, ind ){
if ( typ in tr )
typ = tr[typ]
if ( name ~ /bytes/ ){
optional = "repeated"
}
for (c = 1; c <= tsi; c++){
sub("^\\[\\]" ts[c, 1], "[]" ts[c, 2], typ)
sub("^\\[\\]\\*" ts[c, 1], "[]*" ts[c, 2], typ)
sub("^" ts[c, 1], ts[c, 2], typ)
}
if ( typ ~ /map/ ){
optional = ""
}
if ( typ ~ /^\[\]/ ){
optional = "repeated"
sub(/^\[\]/, "", typ)
}
sub(/^\[.*\]byte$/, "bytes", typ)
sub(/^time.Duration$/, "sint64", typ)
sub(/^kyber.Point$/, "bytes", typ)
sub(/^kyber.Scalar$/, "bytes", typ)
sub(/^int32$/, "sint32", typ)
sub(/^int64$/, "sint64", typ)
sub(/^int$/, "sint32", typ)
sub(/^\*/, "", typ)
print sprintf(" %s %s %s = %d;", optional, typ, tolower(name), ind )
}