Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/CosmWasm/mesh-security-hackathon

Implementation of Sunny's Mesh Security talk (Hackathon / Prototype status)
https://github.com/CosmWasm/mesh-security-hackathon

Last synced: 2 months ago
JSON representation

Implementation of Sunny's Mesh Security talk (Hackathon / Prototype status)

Awesome Lists containing this project

README

        

# mesh-security (Hackathon / Prototype status)

**Note** this is working (but fragile) code from Sep 2022 Medellin HackWasm. Feel free to bang on it to
understand Mesh Security better. As of April 2023, we have started a new repo
[osmosis-labs/mesh-security](https://github.com/osmosis-labs/mesh-security), which will host
work towards a full production-quality implementation of Mesh Security, written by the same
developers and many more.

## What it is

An implementation of Sunny's [Mesh Security](https://youtu.be/Z2ZBKo9-iRs?t=4937) talk from Cosmoverse 2022.

This should run on any CosmWasm enabled chain. This is MVP design and gives people
hands on use, that should work on a testnet. Open questions that need
to be resolved before we can use this in production are listed below.

## Architecture

Please check out the [architectural documentation](./docs/README.md), which gives a thorough view
of all the components in a completed system.

You can also look at [Sunny's original flipboard](https://docs.google.com/presentation/d/13JrSdzodhAGRj59-P5e9sxyoDPvVm7ZVvw1GzR8OilQ/edit#slide=id.g15d37737ae2_0_885)
to get view of the various flows involved.

## Contracts

* `meta-staking` - a bridge between the rest of the contracts and the x/staking module to
provide a consistent, friendly interface for our use case
* `mesh-lockup` - a contract that locks tokens and allows lockers to issue multiple claims
to other consumers, who can all slash that stake and eventually release their claim
* `mesh-provider` - an IBC-enabled contract that issues claims on an ILP and speaks IBC to a consumer. It
is responsible for submitting slashes it receives from the `slasher` to the `ilp` contract.
* `mesh-consumer` - an IBC-enabled contract that receives messages from `ibc-provider` and
communicates with `meta-staking` to update the local delegations / validator power
* `mesh-slasher` - a contract that is authorized by the `mesh-provider` to submit slashes to it. There can
be many types of slasher contracts (for different types of evidenses of misbehaviors)

## Overview for Users

**High Level:** You connect Osmosis to Juno and Juno to Osmosis. We will only look at one side
of this, but each chain is able to be both a consumer and producer at the same time.
You can also connect each chain as a provider to N chains and a consumer from N chains.

Let's analyze the Osmosis side of this. Osmosis is the provider of security to Juno.
Once the contracts have been deployed, a user can interact with this as follows.

#### Cross-staking:

1. User stakes their tokens in the `mesh-lockup` contract on Osmosis
2. User can cross-stake those tokens to a Juno `mesh-provider` contract (on Osmosis), specifying how many of their
tokens to cross-stake and to which validator
3. The Osmosis `mesh-consumer` contract (on Juno) receives a message from the counterparty `mesh-provider` contract
and updates the stake in the `meta-staking` contract (on Juno).
4. The `meta-staking` contract checks the values and updates it's delegations to `x/staking` accordingly. (The
meta-staking contract is assumed to have enough JUNO tokens to do the delegations. How it gets that JUNO is
out of scope.)

#### Claiming Rewards:

1. Anyone can trigger the Osmosis consumer contract to claim rewards from the `meta-staking` contract
2. The `mesh-consumer` contract (on JUNO) sends tokens to the `mesh-provider` contract (on Osmosis) via ics20
3. The `mesh-consumer` contract sends a message to the `mesh-provider` contract to inform
of the new distribution (and how many go to which validator).
4. The `mesh-provider` (on Osmosis) contract updates distribution info to all stakers, allowing them to claim
their share of the $JUNO rewards on Osmosis.

#### Unstaking:

1. A user submits a request to unstake their tokens from the `mesh-provider` contract (on Osmosis)
2. We update the local distribution info to reflect the new amount of tokens staked
3. This sends a message to the `mesh-consumer` contract (on Juno), which updates the Juno `meta-staking` contract
to remove the delegation.
4. The `mesh-provider` contract (on Osmosis) gets the unbonding period for this cross stake by querying
the `slasher` contract
5. After the unbonding period has passed (eg. 2 weeks, 4 weeks) the `mesh-provider` contract
informs the `mesh-lockup` contract that it removes its claim.
6. If the user's stake in the `mesh-lockup` contract has not more claims on it, they can withdraw their stake.

#### Slashing:

1. Someone calls a method to submit evidence of Juno misbehavior on the `meta-slasher` contract (on Osmosis).
2. The `meta-slasher` contract verifies that a slashing event has indeed occurred and makes a contract call to the
`mesh-provider` contract with the amount to slash.
3. The `mesh-provider` updates the `mesh-lockup` stakes of everyone delegating to the offending validator. Tokens are unbonded
and scheduled to be burned.
4. `mesh-provider` sends IBC packet updates to the `mesh-consumer`s on all other chains about the new voting power.

#### Claiming tokens:

A user can stake any number of tokens to the `mesh-lockup` contract, and use them in multiple provider contracts.
The `mesh-lockup` contract ensures that the user has balance >= the max claim at all times.
If you put in eg 1000 OSMO, but then provide 700, 500, and 300 to various providers,
you can pull out 300 OSMO from the `mesh-lockup` contract. Once you successfully release the claim on the
provider with 700, then you can pull out another 200 OSMO.

## Overview for Installing

1. Deploy the contracts to Osmosis and Juno
2. `x/gov` on Juno will tell the "Osmosis consumer contract" which `(connectionId, portId)` to trust
3. `x/gov` on Juno will provide the "Osmosis consumer contract" with some JUNO tokens with which it can later delegate
(This is a hacky solution to give the consumer contract OSMO to be able to stake... we discuss improvement below).
4. A relayer connects the two contracts, the consumer contract ensures that the channel is made
from the authorized `(connectionId, portId)` or rejects it in the channel handshake. It also
ensures only one channel exists at a time.
5. Once the trusted connection is established and the consumer contract has been granted sufficient
delegation power, then the user flow above can be used.

## Features to add

These are well-defined but removed from the MVP for simplicity. We can add them later.

* `mesh-lockup` must also allow local staking, and tie into the meta-staking contract to use that
same stake to provide security on the home chain.

## Open Questions

These are unclear and need to be discussed and resolved further.

* How to cleanly grant consumer contracts the proper delegation power?
* Something like "superfluid staking" module where we can mint "synthetic staking tokens"
that work like normal, but are removed from the "totalSupply" query via offset.
* Fork `x/staking` to allow such synthetic delegations that don't need tokens.
This is a hard lift, but would allow custom logic, like counting those tokens
in tendermint voting power, but exclude them from `x/gov`, and decide on some
reducing factor for their rewards.
* How to cleanly define limits for the providing chains on how much power they can
have on the consuming chain? We start with a fixed number (# of JUNO), but better
to do something like "max 10% of total staking power".
* Ensure a minimum voting power for the local stake. If we let 3 chains each use up to 30%
of the voting power, and they all stake to the max, then we only have 10% of the power locally.
We can set a minimum to say 40% local, and if all remote chains stake to the max, their
relative powers are reduced proportionally to ensure this local minimum stake.
* How to normalize the token values? If we stake 2 million $OSMO, we need to convert that
to the same $$ value of $JUNO before using it to calculate staking power on the Juno chain.
* How to properly handle slashing, especially how a slashing on JUNO triggers a slash on OSMO,
which should then reduce the voting power of the correlated validators on STARS
(that was based on the same OSMO stake). This is a bit tricky, IBC messages could be sent out
to all consumer chains, but there could be performance implications for this.
* Desired reward payout mechanism. For MVP, we treat this as a normal delegator and
send the tokens back to the provider chain to be distributed. But maybe we calculate
rewards in another way, especially when we modify `x/staking`. Should also be computationally
efficient
* How to improve installation UX? Ideally, when the consumer chain votes to instantiate
mesh security with another chain, all contracts are deployed and configured in one governance
prop.
* How to handle changing prices? What is a good price oracle? We need to update the voting
power of the consumer contract when the price changes. This may lead to issues in normalization
especially if the remote token price rises considerably.
* What do we do when the consumer stake is greater than the max allowance? Do we fail that extra Stake?
Do we normalize the validators within that consumer? Failing is easier, but not possible in
response to price oracle changes. For example, if max power is 1000 and we have 500 for val A and 1000
for val B and 1000 for val C, we could normalize to A=200, B=400, C=400.