Skip to content

Commit

Permalink
New text from Michel
Browse files Browse the repository at this point in the history
  • Loading branch information
xeroc committed Aug 7, 2019
1 parent 6bdb11f commit d716db1
Showing 1 changed file with 149 additions and 66 deletions.
215 changes: 149 additions & 66 deletions bsip-0047.md
Original file line number Diff line number Diff line change
@@ -1,99 +1,182 @@
BSIP: <BSIP number>
Title: An explicit voting operation
BSIP: 47
Title: Vote Proxies for Different Referendum Categories
Authors: Fabian Schuh <https://github.com/xeroc>
Dmitrij Vinokour <https://github.com/Dimfred>
Alex Megalokonomos <https://github.com/clockworkgr>
Michel Santos <https://github.com/MichelSantos>
Status: Draft
Type: Protocol
Created: 2018-09-20
Discussion: <url>
Worker: <Id of worker proposal> (optional)
Discussion: https://github.com/bitshares/bsips/pull/114


# Abstract

On the BitShares Blockchain the `account_update_operation` allows to change account
related settings. In the current implementation all possible settings have to be
passed to the operation, to not override old settings. This is inconvenient
because in certain situations there is no need in changing all settings.
Voting power by core token holders may be assigned to different proxies in each of the BitShares referendum categories: witnesses, committee members, and worker proposals.

In this BSIP the `account_update_votes_operation` is introduced, which solves the
described problem.

# Motivation

The current `account_update_operation` forces the user to pass all possible account
settings, to not override existing ones. This is inconvinient and also requires a
higher payload to be stored on the blockchain, thus results in a higher fee.
Some proxies can be more or less knowledgeable, wise, and trusted by token holders to vote on certain referendum categories than others. It is therefore desirable to empower token holders to select different proxies for each of the referendum categories.

If the ability to select multiple proxies or to directly vote through a single new operation were made possible, this new operation be used in conjunction with [BSIP40](https://github.com/bitshares/bsips/blob/master/bsip-0040.md) which will allow an account holder to assign a specific key for voting. This combination could simplify voting for many account holders.


# Rationale

To reduce the fee the `account_update_votes_operation` is introduced to update only
a part of the `account data`.
Holders of the core token of BitShares (BTS) have always had the ability to directly vote on three referendum categories: witnesses, committee members, and worker proposals. They have also had the option of delegating their voting power to another account (a "proxy") to vote on any of these referendum categories.

This new operation will be particularily useful in conjunction with
[BSIP40](https://github.com/bitshares/bsips/blob/master/bsip-0040.md)
which will allow to assign a specific key for this operation and
simplifies life for voters, significantly.
Rather than delegating voting power to a single proxy for each of three referendum, a core token holder might prefer to select different proxies for each of the referendum categories.

# Rational
This more granular approach empowers core token holders with more options. A BTS token holder may choose to directly vote on any referendum category including the option to abstain. A BTS token holder may choose to select proxies for each of the referendum categories. Each of the referendum categories may have distinct proxies or may have common proxies. Some referendum categories may have assigned proxies while others may be unassigned. _Any referendum category that does not have an assigned proxy permits the voter to vote directly on any referendum in that category._ _But if a **confused** core token holder votes for a referendum in a category **while also** delegating that category to a proxy, the token holder's direct vote shall be ignored._

The `account_update_votes_operation` is an account related operation and allows to set
variables in the `account_object`. The targeted variables of this operation are located
in the `account_options field` of the `account_object`.
The introduction of a new operation (`account_update_votes_operation`) with additional changes to the core shall also enable an account holder to select proxies for different referendum categories, and to vote for referendums without needing to use the existing `account_update_operation`. The existing operation requires the account holder to pass all possible account settings to avoid overriding existing settings. This is inconvinient and requires a higher payload to be stored on the blockchain which results in the account holder paying a higher fee. The proposed operation shall only update the portion of the `account data` that is related to voting and should also reduce the fee.

- `voting_account` (the voting account to use)
- `num_witness` (number of witness votes)
- `num_committee` (number of committee votes)
- `votes` (list of `vote_id_type`)
The new operation only needs to include

- The voting account
- The desired proxy, if any, for the _committee_ referendum category
- The desired proxy, if any, for the _witness_ referendum category
- The desired proxy, if any, for the _worker_ referendum category
- The quantity of desired witnesses
- The quantity of desired committee members
- The list of referendums that are supported

As described before to not override existing settings, every other needed variable is fetched
with the database, when the operation is evaluated.

# Specifications

This BSIP comes with a change to the protocol, which requires a propotocol upgrade.

In the following an exemplary implementation is given.

struct account_update_votes_operation : public base_operation
{
asset fee;
/// The account to update
account_id_type account;
/// New account options
flat_set<vote_id_type> votes_to_add;
flat_set<vote_id_type> votes_to_remove;
// a new voting account to set
optional<account_id_type> voting_account;
// a new number of witness_votes to set
optional<uint16_t> num_witness;
// a new number of commitee_votes to set
optional<uint16_t> num_committee;
// for future extensions (see account_update_operation)
extensions_type extensions;
};


We propose to allow adding and removing votes individually by means of
defining a *list* of vote ids to be added or removed. This has multiple
advantages:

* Reduces the size of the operation compared to replacing the entire
slate
Whenever a proxy for a category is _assigned_, any direct votes by a core token holder within that category shall be _ignored_. Whenever a proxy for a category _is unassigned_, any direct votes by a core token holder for that category _shall be counted_ during the vote tallying process of the blockchain. **This shall require an upgrade to the protocol.**

### `GRAPHENE_PROXY_PER_CATEGORY_ACCOUNT` (New)

A new special account `GRAPHENE_PROXY_PER_CATEGORY_ACCOUNT` with ID `1.2.6` is defined and added to the database at protocol change activation time.


### `account_options` extension (Existing)

**Fields:**

The `account_options` receives three new extensions:

* `optional<account_id_type> committee_voting_account`
* `optional<account_id_type> witness_voting_account`
* `optional<account_id_type> worker_voting_account`

**Validation:**

* These extensions must not be present before the protocol change activation date, neither in transactions nor in proposals.
* If any extension is present, it must contain the ID of an existing account.
* If any extension is present, `account_options.voting_account == GRAPHENE_PROXY_PER_CATEGORY_ACCOUNT`


### `account_object`(Existing)

**Note:** this is an implementation detail and meant to be a hint for later development. The actual implementation may differ.

The account object shall receive these new fields:

* `flat_set<vote_id_type> committee_votes`
* `flat_set<vote_id_type> witness_votes`
* `flat_set<vote_id_type> worker_votes`

`account_create_operation` and `account_update_operation` shall be modified to create these sets by filtering the `account_options.votes` field appropriately. `account_update_votes_operation` shall also create these sets by filtering the `votes_to_add` and `votes_to_remove` fields appropriately.


### `account_update_votes_operation` (New)

The following is an example implementation.

```
struct account_update_votes_operation : public base_operation
{
asset fee;
/// The account to update
account_id_type account;
/// New account options
flat_set<vote_id_type> votes_to_add;
flat_set<vote_id_type> votes_to_remove;
// A new voting account to set
optional<account_id_type> committee_voting_account
optional<account_id_type> witness_voting_account
optional<account_id_type> worker_voting_account
// A new number of witness_votes to set
optional<uint16_t> num_witness;
// A new number of committee_votes to set
optional<uint16_t> num_committee;
// For future extensions (see account_update_operation)
extensions_type extensions;
};
```

Changes to direct votes can be specified by means of a *list* of vote identifiers to be added or removed. This has multiple advantages:

* Reduces the size of the operation compared to replacing the entire slate
* Allows to easily compare votes before and after the operation

Setting the proxy is explicitly not part of this opertation.
Authors of this BSIP came to the conclusion that proxy settings should
require `active authority`. It is their opinion that setting proxy
settings are distinct from voting.
**Validation checks:**

* `account` must be a valid account ID, must exist, must authorize the operation, must pay fee
* `committee_voting_account`, if present, must be a valid account ID, must exist
* `witness_voting_account`, if present, must be a valid account ID, must exist
* `worker_voting_account`, if present, must be a valid account ID, must exist
* Validate the existence of each vote to add and remove
* `num_witness`, if present, must be greater than or equal to &ge; 0
* `num_committee`, if present, must be greater than or equal to &ge; 0

**Evaluation:**

* Update the account's `account_options.committee_votes`, `account_options.witness_votes`, and `account_options.worker_votes` by filtering the `votes_to_add` and `votes_to_remove` field appropriately
* Update the `account_options.num_witness` if `witness_voting_account = GRAPHENE_PROXY_TO_SELF_ACCOUNT` and if `num_witness` is present
* Update the `account_options.num_committee` if `committee_voting_account = GRAPHENE_PROXY_TO_SELF_ACCOUNT` and if `num_committee` is present
* Update the account's last vote time to the head block time


### `account_create_operation` (Existing)

This operation shall also be updated for backwards compatibility. If `account_options.voting_account == GRAPHENE_PROXY_PER_CATEGORY_ACCOUNT`, then the account's per-category voting proxies are set to the specified extension if present, or to `GRAPHENE_PROXY_TO_SELF_ACCOUNT` if absent.
Otherwise, the voting proxy for all three categories is set to `account_options.voting_account`.


### `account_update_operation`(Existing)

This operation shall also be updated for backwards compatibility. If `account_options.voting_account == GRAPHENE_PROXY_PER_CATEGORY_ACCOUNT`, then the account's per-category voting proxies are set to the specified extension if present, or remain unchanged if absent.
Otherwise, the voting proxy for all three categories is set to `account_options.voting_account`.


### Vote Tallying

The vote tallying algorithm shall be modified to select the `opinion_account` per voting category, and use the current account's voting stake only for the opinion account's votes in the respective category.

Other BSIPs that may scale the voting weight of an account, such as by [vote decay](https://github.com/bitshares/bsips/bsip-0022.md) or by [vote staking](https://github.com/bitshares/bsips/bsip-0024.md), can remain compatible with this BSIP by assigning the scaled voting weight to the different voting proxies that may or may not be selected by an account.


# Discussion

See BSIP issues on github.
### Performance Impact

There will be a performance impact by `account_create_operation`, `account_update_operation`, and `account_update_votes_operation` due to the filtering if implemented as suggested. This impact should be small overall, because these operations are comparatively rare.

There will also be some overhead for vote tallying. Vote tallying happens only during the maintenance interval so the operational impact should be negligible as well.

Note that the cumulative effect of both may be noticable during chain replay.

# Summary for Shareholders

This BSIP simplifies the way voting occurs on the blockchain and will
later (together with BSIP40) simplify automated voting without
compromising the account.
This proposal introduces a means for token holders to select different proxies in each of the BitShares referendum categories: witnesses, committee members, and worker proposals. The proposed mechanism in combination with [BSIP40](https://github.com/bitshares/bsips/blob/master/bsip-0040.md) will permit core token holders to simplify automated voting without compromising the account through the use of a new operation that can be used only for voting.


# Copyright

This document is placed in the public domain.


# See Also

- [Vote Proxies for Different Referendum Categories](https://github.com/bitshares/bsips/issues/177)
- [Proposed details for BSIP-22 specifications](https://github.com/bitshares/bsips/pull/153)

0 comments on commit d716db1

Please sign in to comment.