aboutsummaryrefslogtreecommitdiff
path: root/spec/srv-spec/protocol.md
blob: 67aa493195fe7702b166a38deb32d24b25980525 (plain)
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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
<a id="srv-spec.txt-3"></a>

# Protocol

In this section we give a detailed specification of the protocol. We
describe the protocol participants' logic and the messages they send. The
encoding of the messages is specified in the next section (\[SPEC\]).

Now we go through the phases of the protocol:

<a id="srv-spec.txt-3.1"></a>

## Commitment Phase {#COMMITMENTPHASE}

The commit phase lasts from 00:00UTC to 12:00UTC.

During this phase, an authority commits a value in its vote and
saves it to the permanent state as well.

Authorities also save any received authoritative commits by other authorities
in their permanent state. We call a commit by Alice "authoritative" if it was
included in Alice's vote.

<a id="srv-spec.txt-3.1.1"></a>

### Voting During Commitment Phase {#commitment-voting}

During the commit phase, each authority includes in its votes:

```text
    - The commitment value for this protocol run.
    - Any authoritative commitments received from other authorities.
    - The two previous shared random values produced by the protocol (if any).
```

The commit phase lasts for 12 hours, so authorities have multiple chances to
commit their values. An authority MUST NOT commit a second value during a
subsequent round of the commit phase.

If an authority publishes a second commitment value in the same commit
phase, only the first commitment should be taken in account by other
authorities. Any subsequent commitments MUST be ignored.

<a id="srv-spec.txt-3.1.2"></a>

### Persistent State During Commitment Phase {#STATECOMMIT}

During the commitment phase, authorities save in their persistent state the
authoritative commits they have received from each authority. Only one commit
per authority must be considered trusted and active at a given time.

<a id="srv-spec.txt-3.2"></a>

## Reveal Phase {#reveal-phase}

The reveal phase lasts from 12:00UTC to 00:00UTC.

Now that the commitments have been agreed on, it's time for authorities to
reveal their random values.

<a id="srv-spec.txt-3.2.1"></a>

### Voting During Reveal Phase {#reveal-voting}

During the reveal phase, each authority includes in its votes:

```text
    - Its reveal value that was previously committed in the commit phase.
    - All the commitments and reveals received from other authorities.
    - The two previous shared random values produced by the protocol (if any).
```

The set of commitments have been decided during the commitment
phase and must remain the same. If an authority tries to change its
commitment during the reveal phase or introduce a new commitment,
the new commitment MUST be ignored.

<a id="srv-spec.txt-3.2.2"></a>

### Persistent State During Reveal Phase {#STATEREVEAL}

During the reveal phase, authorities keep the authoritative commits from the
commit phase in their persistent state. They also save any received reveals
that correspond to authoritative commits and are valid (as specified in
\[VALIDATEVALUES\]).

An authority that just received a reveal value from another authority's vote,
MUST wait till the next voting round before including that reveal value in
its votes.

<a id="srv-spec.txt-3.3"></a>

## Shared Random Value Calculation At 00:00UTC {#midnight-utc}

Finally, at 00:00UTC every day, authorities compute a fresh shared random
value and this value must be added to the consensus so clients can use it.

Authorities calculate the shared random value using the reveal values in
their state as specified in subsection \[SRCALC\].

Authorities at 00:00UTC start including this new shared random value in
their votes, replacing the one from two protocol runs ago. Authorities also
start including this new shared random value in the consensus as well.

Apart from that, authorities at 00:00UTC proceed voting normally as they
would in the first round of the commitment phase (section \[COMMITMENTPHASE\]).

<a id="srv-spec.txt-3.3.1"></a>

### Shared Randomness Calculation {#SRCALC}

An authority that wants to derive the shared random value SRV, should use
the appropriate reveal values for that time period and calculate SRV as
follows.

HASHED_REVEALS = H(ID_a | R_a | ID_b | R_b | ..)

```text
      SRV = SHA3-256("shared-random" | INT_8(REVEAL_NUM) | INT_4(VERSION) |
                     HASHED_REVEALS | PREVIOUS_SRV)
```

where the ID_a value is the identity key fingerprint of authority 'a' and R_a
is the corresponding reveal value of that authority for the current period.

Also, REVEAL_NUM is the number of revealed values in this construction,
VERSION is the protocol version number and PREVIOUS_SRV is the previous
shared random value. If no previous shared random value is known, then
PREVIOUS_SRV is set to 32 NUL (\\x00) bytes.

To maintain consistent ordering in HASHED_REVEALS, all the ID_a | R_a pairs
are ordered based on the R_a value in ascending order.

<a id="srv-spec.txt-3.4"></a>

## Bootstrapping Procedure {#bootstrapping}

As described in \[CONS\], two shared random values are required for the HSDir
overlay periods to work properly as specified in proposal 224. Hence
clients MUST NOT use the randomness of this system till it has bootstrapped
completely; that is, until two shared random values are included in a
consensus. This should happen after three 00:00UTC consensuses have been
produced, which takes 48 hours.

<a id="srv-spec.txt-3.5"></a>

## Rebooting Directory Authorities {#REBOOT}

The shared randomness protocol must be able to support directory
authorities who leave or join in the middle of the protocol execution.

An authority that commits in the Commitment Phase and then leaves MUST have
stored its reveal value on disk so that it continues participating in the
protocol if it returns before or during the Reveal Phase. The reveal value
MUST be stored timestamped to avoid sending it on wrong protocol runs.

An authority that misses the Commitment Phase cannot commit anymore, so it's
unable to participate in the protocol for that run. Same goes for an
authority that misses the Reveal phase. Authorities who do not participate in
the protocol SHOULD still carry commits and reveals of others in their vote.

Finally, authorities MUST implement their persistent state in such a way that they
will never commit two different values in the same protocol run, even if they
have to reboot in the middle (assuming that their persistent state file is
kept). A suggested way to structure the persistent state is found at \[STATEFORMAT\].