SubstrateでDemocracy(民主主義)機能を試してみる


はじめに

PolkadotではDemocracy(民主主義)という機能がありますが、提案(あるアカウントに運営側から100トークン付与する、など)をして各アカウントがその提案に対して、賛成か反対かを投票することができ、可決されればその提案内容を実行する事ができたりします。

また評議会というものがあり、評議会のメンバーになると提案を優先的に可決させたり、否決させたりする事ができます。

今回はこのDemocracy機能をSubstrate node templateに導入して動作確認をしてみたいと思います。

導入手順

記事のソース

pallet-democracyの導入

pallet-democracyを導入します。
こちらのpalletを導入することで、提案する事ができ、その提案に対する投票ができるようになります。

Cargo.toml

こちらのpalletを導入するにあたり関連のpalletもCargo.tomlに導入する事になります。

実際に導入するpalletはこちらを参照してください。

node/src/chain_spec.rs

pallet追加に伴うchain_specのソース修正を行います。

testnet の genesis config 設定で、democracy, council, technical_committee, treasury を追加します。

runtime/src/constants.rs

Substrateソースであるconstants.rsをコピーして runtime/src/constants.rs とします。

runtime/src/lib.rs

追加したpallet毎に、parameter_types、pallet_XXXXX::Config を設定します。
ちなみにこの時のConfig設定で投票期間などの時間系も設定できます。

追加内容はこちら

試験用に投票期間などを短くした修正はこちらです。

	pub const LaunchPeriod: BlockNumber = 6 * MINUTES;  // 投票受付のサイクル
	pub const VotingPeriod: BlockNumber = 6 * MINUTES;  // 投票期間
	pub const FastTrackVotingPeriod: BlockNumber = 1 * MINUTES;
	pub const EnactmentPeriod: BlockNumber = 7 * MINUTES;  // 提案可決後に提案実行するまでの期間
	pub const CooloffPeriod: BlockNumber = 6 * MINUTES;  // 拒否された後に外部提案を再提出できない期間

Council(評議会)対応

pallet-democracy関連のpalletを導入しただけですと(pallet-democracyを導入してcompileを通しただけ)、提案&提案に対する投票はできますがCouncil(評議会)へのメンバー参加や、そのメンバーになれるかどうかの投票ができません。

このためpallet-elections-phragmenを導入します。

修正内容はこちらです

確認手順

提案(プロポーザル)作成から提案実行まで

提案(プロポーザル)をするには[preimage hash]と呼ばれるものを送信する必要があります。
[preimage hash]は登録する提案のhashです。

大きな[preimage hash]を提出するための保管コストはかなり高くつく可能性があるため、提案を作成する行為と、提案の[preimage hash]を提出することと分けています。
[preimage hash]の送信を別のトランザクションとして許可するということは、資金がない場合に別のアカウントが[preimage hash]を送信できることを意味します。

では、この[preimage hash]を作成するために[Submit preimage]ボタンを押します。

画面上で提案内容を入力して(下の画面は FERDIEの残高を設定するという提案)[preimage hash]を覚えておき、画面の指示にしたがいます。

今度は[Submit proposal]ボタンを押します。

そして前回の画面でコピーしておいた[preimage hash]を入力して[Submit proposal]ボタンを押します。

無事Submitするとproposalsセクションに提案内容が表示されます。

[launch period]の期間が過ぎると(画面上は6分です)、以下のように proposals から referenda(国民投票) に移動し Vote(投票) できるようになります。

ALICEでこの提案に対してYES[Aye]でVoteしてみます。
この際以下画面の[vote value(投票に使うトークン量)]と[conviction(ロックする期間)]によって[vote power]が変わってきます。詳しくはこちらを参照してください。

すると以下のように緑チェックマークがついてAye(YES)にVoteした量が表示されています。

Bobで否定してみます。Aliceより少なくなるように[vote value][conviction]を入力して[Vote Nay]ボタンを押します。

Ayeの方が多いので、可決予定のままです。
ただ、この辺りの[vote power]についてはこちらに詳しく記載があります。

そして[Launch period]の期間がすぎるとDispatchのqueueにたまります。
以下画面ですと enact, remaining が6分36秒後に実施する事になっています。
これは EnactmentPeriodを7分として設定しているため24秒経過している状態です。

7分経過後、以下のように提案が実行されて FERDIE を見ると 100 UNIT 振り込まれてる事が確認できました。

評議会のメンバー追加

冒頭でも軽く紹介しましたが評議会では、評議会のメンバーになると提案を優先的に可決させたり、否決させたりする事ができます。

こちらの評議会のメンバーになるには上記タブメニューから Governance - カウンシル を選択したあと[Submit candidacy]を押します。

この後、評議会に推薦するアカウントを指名することになりますが、推薦するアカウントの残高をかなり多めに持たせておかないと、推薦できないので注意が必要です。
今回は 10000000000000000000 を付与しておきました。

[submit your council candidacy]画面で推薦するアカウントを選択後に[Submit]ボタンを押します。

すると[candidates]セクションに、推薦したアカウントが表示されます。

candidateしただけだと、メンバーになることはできず、このあとVoteする必要があります。

Voteボタンを押して[Vote value]、投票するアカウントを決めてVoteします。これで[member]もしくは[runners up]のセクションに[term processes]の期間を過ぎると移動します。

今回、すでに評議会メンバーである Alice と Bob の backing が 1MUNIT(デフォルト)で、先ほどVoteしたvalueが足りないため[runners up]にまわされます。

推薦したアカウントを評議会メンバーにしたいので[Vote]を改めて行い[vote power]を増やす事によって既存のメンバーと入れ替える事ができます。

これで評議会メンバーに新たなアカウントを入れる事ができました。

終わりに

今回は Substrate の Democracy(民主主義)機能 について触れてみました。

運営側で強制的に何かしらの処理をSudo機能で実行もできますが、こういった全アカウント対象として投票制で処理を実行できるかどうかを決めれる機能が用意されてるブロックチェーンフレームワークもそうないのではないでしょうか(あったらすみません)。

Substrate ではこういった機能がpalletを追加する事によって動作させる事が可能で非常に使い勝手が良いと考えています。(palletによっては追加が多少面倒な時もありますが)
今後も他に気になる機能があったら記事として紹介していこうと思います!