shortname: 3/UPSERT-VALIDATORS
name: Dynamically add/update/remove validators at runtime
type: standard
status: stable
editor: Vanshdeep Singh <[email protected]>
In the current state of tendermint bdb, validator nodes are specified in the Tendermint config file before starting the node. It is necessary that the users should be able to dynamically add new validators.
NOTE: This document presents an implementation intended for BigchainDB 2.0.
Tendermint allows the abci client application to return a list of validators during end_block
call. If the client doesn't wish to modify the validators an empty list should be returned. The return value allows the client to dynamically add/remove validators. Note that the return value could be thought of as a (validators) diff of desired validators list and current validators list. Below is an example of validators diff,
[{"pub_key":{"type":"ed25519","data":"4E2685D9016126864733225BE00F005515200727FBAB1312FC78C8B76831255A"},
"power":10},
{"pub_key":{"type":"ed25519","data":"608D839D7100466D6BA6BE79C320F8B81DE93CFAA58CF9768CF921C6371F2553"},
"power":0}
]
Note: "power": 0
implies that the validator should be removed from the validator list. In case this diff is empty the validators remain same. Furthermore, the validators returned during end_block
are used to update the validators of the given Tendermint node and not for the whole network.
-
Add a new subcommand
bigchaindb upsert-validator PUBLIC_KEY POWER
which will allow the node operator to add/update/delete a validator. -
GET /api/v1/validators
should be introduced to list the current validators.
Since node operator can call upsert-validator
anytime it is necessary to (temporarily) store the payload until end_block
is called. This implies that just executing bigchaindb upsert-validator
doesn't ensure that the validator is add to validator's set.
Furthermore, as discussed in technical details each/majority (>2/3) nodes in the network need to add the new node to their list of validators in order for the new node to be able to act as a validator. Until then the node may not be able to participate and vote on blocks.
In the previous section it was discussed that it is necessary to (temporarily) store the updates to validators list until end_block
is executed. Following is one possible solution,
- Store a validator update requested by the node operator in a MongoDB collection
validators
,
{"validator": {"pub_key":{"type":"ed25519","data":"4E2685D9016126864733225BE00F005515200727FBAB1312FC78C8B76831255A"},
"power":10},
"update_id": "validator01"
}
Note, the node operator is allowed to add only one validator at a time i.e. until the current requested validator is added to the validator's set, the node operator is not allowed to submit another request for updating the validator's set.
- The value of primary field
"update_id"
is always"validator01"
. This ensures that no new validator requests can be added to the collection until the current validator request has been processed and deleted from the collection. - During
end_block
the document with"update_id": "validator01"
is fetched and validator diff list is prepared which is returned inend_block
response.
NOTE: A validators diff which results in a change of >1/3
of voting power will be rejected by Tendermint and an error log message will be generated in Tendermint's console.
-
A new subcommand
upsert-validator
should be introduced to add/update/delete validators- syntax :
bichaindb upsert-validator PUBLIC_KEY POWER
- example usage
- add/update validator
$ bichaindb upsert-validator 4E2685D9016126864733225BE00F005515200727FBAB1312FC78C8B76831255A 10
- delete validator
$ bichaindb upsert-validator 4E2685D9016126864733225BE00F005515200727FBAB1312FC78C8B76831255A 0
- syntax :
-
List validators
- Uri:
GET /api/v1/validators
- response:
[{"pub_key":{"type":"ed25519","data":"4E2685D9016126864733225BE00F005515200727FBAB1312FC78C8B76831255A"}, "power":10}, {"pub_key":{"type":"ed25519","data":"608D839D7100466D6BA6BE79C320F8B81DE93CFAA58CF9768CF921C6371F2553"}, "power":0} ]
- Uri:
NOTE: when power: 0
is set for a given validator it is removed from the validators list.
Below is a summary of workflow sequence which should be executed in order to add/remove a validator dynamically,
- BigchainDB node operator wishes to add/update/delete a validator from their own node.
- The operator uses
bigchaindb upsert-validator
to submit a validator add/update/delete request, - BigchainDB stores the payload in a MongoDB collection as follows,
{"validator": {"pub_key":{"type":"ed25519","data":"4E2685D9016126864733225BE00F005515200727FBAB1312FC78C8B76831255A"},
"power":10},
"update_id": "validator01"
}
- Now, when
end_block
is executed, the validator request from MongoDB collection is fetched. - A validators diff is prepared are returned during
end_block
and the validator update is deleted from thevalidators
collection in MongoDB.
The current implementation is somewhat insecure because anyone who gains access to a node can modify its local list of validators. However, unless >2/3 of the nodes make the same change, it won't take effect.
The new api introduced should be documented along with its current state of security.
Following test cases should be included:
- Add a new validator
- Remove a validator
- Change the power of a validator
NOTE: The above tests should check if the MongoDB collection is updated and calling
end_block
returns a valid diff. More accurate tests could be written in the integration test suite whereinhttp://tendermint:46657/validators
is called to verify the validator changes.
Primary assignee(s): @kansi
BigchainDB 2.0
To the extent possible under law, the person who associated CC0 with this work has waived all copyright and related or neighboring rights to this work.