Background

On Sept 5 and Sept 7, multiple validators were slashed by the oracle module. As far as we can tell, the validators were running a legitimate version of the pricefeeder program and were not acting maliciously, but because of recent market volatility on ATOM:USD, these validators were slashed.

This report aims to explain how the oracle module works and present findings from the slashing investigation.

Oracle Mechanics

A new vote period begins every N blocks (N=30). The EndBlocker hook of the block right before a new period aggregates price votes. For example, if a new vote period begins on block 90, then the prices are aggregated at the end of block 89.

Pricefeeder listens to new blocks and send a price vote transaction after the block before a new voting period is committed. For example, once block 89 is published, the pricefeeder will react and immediately send a price vote. These price votes may be included in block 90 or a later block, depending on the tx propagation and network latency, but as long as they are included in block 119, then it’ll count as a vote for the period.

Slashing Mechanics

A validator must make a minimum number of valid votes to prevent slashing. The current min_valid_per_window is 69% and a slash window is 3600 blocks. Given a vote period of 30 blocks, that’s 120 vote periods per slash window. In order to meet the 69% requirement, their miss budget is 31% = 37.2 misses. If a validator misses more than 37 times, they are slashed 0.5% of their stake.

Oracle Params

$ nibid q oracle params | jq

{
  "params": {
    "vote_period": "30",
    "vote_threshold": "0.333333333333333333",
    "reward_band": "0.020000000000000000",
    "whitelist": [
      "ubtc:uusd",
      "ueth:uusd",
      "uatom:uusd",
      "uusdc:uusd",
      "uusdt:uusd",
      "unibi:uusd"
    ],
    "slash_fraction": "0.005000000000000000",
    "slash_window": "3600",
    "min_valid_per_window": "0.690000000000000000",
    "twap_lookback_window": "900s",
    "min_voters": "4",
    "validator_fee_ratio": "0.050000000000000000",
    "expiration_blocks": "900"
  }
}

Slashings

In total, there were three slashings:

Start block End block Slash timestamp
First slashing 11088000 11091599 2024-09-05 01:38:52.892951 +00:00
Second slashing 11196000 11199599 2024-09-07 12:26:14.423507 +00:00
Third slashing 11199600 11203199 2024-09-07 14:23:02.215756 +00:00

First slashing

Validator address Slashed amount (NIBI) Miss Count Missed Price Pair
nibivaloper1he44du3wsg3qgysyj0wlclq20apj880k72m8ks 6938.990000 40 unibi:uusd
nibivaloper1sxpnk6jja5ca6jxe4nf5c7v0zq3tl5jc9fp97y 95.200000 41 unibi:uusd
nibivaloper1waavtr8qpnqalrnzfzjcn93u6gz46hgcsjvmzm 3132.045000 42 unibi:uusd
nibivaloper1wu3yd9xkhetcs7zjtz0y0pt8kh6dm0r37fyzyf 119.150000 43 unibi:uusd

Second slashing

Validator address Slashed amount (NIBI) Miss Count Missed Price Pair
nibivaloper12l4t7ywa35ugq00jzkczwylqvphdk9apxjnacq 9354.120000 52 uatom:uusd
nibivaloper13ajj3qtu49cy8xj3thycal3m4qr0lfkjrxfm9s 9.475000 43
1 uatom:uusd
unibi:uusd
nibivaloper19kmhwcazdjan9cuxnc9t2u6tzlpfzq82rzpx7s 642.930000 48 uatom:uusd
nibivaloper1ej2es5fjztqjcd4pwa0zyvaevtjd2y5wavljes 3166.040000 50 uatom:uusd
nibivaloper1s504sp9yelhws2saw4rq03dh29yvna22ycnfhm 7609.745000 56 uatom:uusd
nibivaloper1w26kzhwhely77xup3npfh70tzuc4amtx8j0743 7260.425000 43 uatom:uusd
nibivaloper1waavtr8qpnqalrnzfzjcn93u6gz46hgcsjvmzm 3134.890000 50 uatom:uusd