This repository has been archived by the owner on Nov 23, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
20 changed files
with
533 additions
and
55 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
--- | ||
This page describes the AttestationStation smart contract and how its used within the EigenTrust protocol context. | ||
--- | ||
|
||
# AttestationStation | ||
|
||
The AttestationStation smart contract is a key component of the EigenTrust protocol, which is designed to enable trust among peers in decentralized networks. Attestations, also known as opinions or ratings, are an important part of the protocol as they allow peers to express their trust or distrust of other peers. The AttestationStation contract serves as a repository for attestations submitted by peers in the network. | ||
|
||
Examples (Let's assume we are doing ratings from 0-5): | ||
- Alice attests Bob with a rating of 5 | ||
- Alice attests to Carol with a rating of 2 | ||
- Bob attests to Alice with a rating of 3 | ||
- Carol says Bob with a rating of 4 | ||
- Alice attests Bob with a rating of 1 | ||
|
||
The mapping attestations is a 3-dimensional mapping that stores attestations submitted by peers. The first key is the address of the peer submitting the attestation. The second key is the address of the peer being attested to. The third key is a bytes32 hash representing the transaction or interaction between the peers. The value stored in the mapping is a bytes array representing the attestation data. | ||
```solidity | ||
mapping(address => mapping(address => mapping(bytes32 => bytes))) public attestations; | ||
``` | ||
|
||
The AttestationData struct represents an attestation submitted by a peer. The `about` field is the address of the peer being attested to. The `key` field is a bytes32 hash representing the transaction or interaction between the peers. The `val` field is a bytes array representing the attestation data. The structure of the `val` field is described in more detail in the [Attestations](../1_attestations.md) documentation. | ||
```solidity | ||
struct AttestationData { | ||
address about; | ||
bytes32 key; | ||
bytes val; | ||
} | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,68 @@ | ||
--- | ||
This page describes the smart contract used for attestations | ||
--- | ||
|
||
# Attestations | ||
|
||
Attestations are the ratings or opinions given by one peer about another peer in the EigenTrust protocol. Each attestation is given for a single transaction or interaction between peers. | ||
|
||
The structure of an attestation is defined as follows: | ||
```rust | ||
struct Attestation<F: FieldExt> { | ||
about: F, | ||
key: F, | ||
value: F, | ||
message: F | ||
} | ||
``` | ||
|
||
Here's a breakdown of each field in the attestation: | ||
|
||
- `about`: the Ethereum address of the peer being rated. This could be an EOA, a smart contract, a DAO, etc. | ||
- `key`: a unique identifier for the transaction or interaction being rated. This could be a hash of the transaction data or a random number generated by the rater. | ||
- `value`: the score given by the rater for the transaction or interaction. The score can range from 0 to a maximum score defined as a constant in the protocol. | ||
- `message`: an optional field for attaching additional information to the attestation. This could be a message from the rater, a domain in which the transaction took place, or a content hash related to the transaction. | ||
|
||
To ensure the integrity and authenticity of an attestation, it is hashed using the Poseidon hash function and then signed using the ECDSA signing algorithm: | ||
```rust | ||
let att_hash = Poseidon::hash(attestation); | ||
let sig = ECDSA::sign(att_hash, keys); | ||
``` | ||
The resulting signature, value and message bytes are stored in the AttestationStation smart contract. The bytes layout would be: | ||
```rust | ||
r = [u8; 32] | ||
s = [u8; 32] | ||
value = u8 | ||
message = [u8; 32] | ||
``` | ||
This adds up to 97 bytes or 65 if we exclude message bytes. | ||
|
||
In case of fetching the attestation from AS and verifying it - first, we read the event: | ||
```solidity | ||
event AttestationCreated( | ||
address indexed creator, | ||
address indexed about, | ||
bytes32 indexed key, | ||
bytes val | ||
); | ||
``` | ||
|
||
Using this data, we extract the `r` and `s`, we verify the signature: | ||
```rust | ||
let (r, s, value, message) = extract_r_s_value_message(val); | ||
let att = Attestation::new(about, key, value, message); | ||
let hash = Poseidon::hash(att); | ||
let is_valid = ECDSA::verify(pub_key, r, s, hash); | ||
assert!(is_valid); | ||
``` | ||
|
||
Then check if the used `pub_key` is actually the pre-image of the `creator`: | ||
```rust | ||
let pk_hash = keccak256(pub_key); | ||
let creator_address = to_address(pk_hash); | ||
assert!(creator_address == creator); | ||
``` | ||
|
||
See [AttestationStation](../0_attestation_station.md) for more details on how attestations are stored and managed in the EigenTrust protocol. | ||
|
||
By signing the attestation, the rater can prove that they made the rating and that the rating has not been tampered with. This is important for verifying the validity of the attestation in an off-chain environment, such as when calculating the EigenTrust scores for each peer. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
--- | ||
This page provides an overview of how data is processed after being pulled from AttestationStation. The data stored in AttestationStation is submitted from one user to another, but the EigenTrust algorithm processes opinions from one user to the whole group. | ||
--- | ||
|
||
To start, a group needs to be defined. This is typically done by assigning a group ID and creating a list of peers, as shown below: | ||
``` | ||
group_id = 1377 | ||
group = [peer1, peer2, peer3, peer4, peer5] | ||
``` | ||
|
||
|
||
And lets assume we have some attestations in AS already: | ||
``` | ||
peer1 => peer2 => 1377 => 5 | ||
peer2 => peer3 => 1377 => 7 | ||
peer4 => peer2 => 1377 => 3 | ||
``` | ||
|
||
Once the group is defined, the next step is to search AttestationStation to construct an opinion map. This is done by iterating through each peer in the group and searching for relevant attestations. The pseudo code snippet below demonstrates this process: | ||
```rust | ||
for i in 0..group.len() { | ||
let peer_i = group[i]; | ||
if peer_i == null { | ||
continue; | ||
} | ||
for j in 0..group.len() { | ||
let peer_j = group[j]; | ||
|
||
let is_null = peer_j == null; | ||
let is_self = peer_i == peer_j; | ||
if is_null || is_self { | ||
continue; | ||
} | ||
|
||
let att = AS.attestations(peer_i, peer_j, group_id); | ||
op_map[peer_i][j] = (peer_j, att); | ||
} | ||
} | ||
``` | ||
|
||
In this code, `peer_i` and `peer_j` represent two peers in the group, and `AS.attestations(peer_i, peer_j, group_id)` retrieves the attestation between the two peers for the given group ID. The resulting opinion map, stored in the op_map variable, is a two-dimensional array that maps each peer to a list of their attestations with other peers in the group. | ||
|
||
Here's an example of what the opinion map might look like, based on the attestations shown earlier: | ||
``` | ||
peer1_op => [(peer1, 0), (peer2, 5), (peer3, 0), (peer4, 0), (peer5, 0)] | ||
peer2_op => [(peer1, 0), (peer2, 0), (peer3, 7), (peer4, 0), (peer5, 0)] | ||
peer4_op => [(peer1, 0), (peer2, 3), (peer3, 0), (peer4, 0), (peer5, 0)] | ||
``` | ||
|
||
This opinion map is then passed to a filtering algorithm before being passed to the EigenTrust algorithm.\ | ||
The details of the filtering algorithm are discussed in more detail in the [Dynamic Sets](../3_dynamic_sets.md) page. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,110 @@ | ||
--- | ||
This page describes our dynamic sets filtering algorithm. | ||
--- | ||
|
||
Suppose we create a fixed set of peers with a limit of 5.\ | ||
Instead of using Ethereum addresses we will use simplified identifiers | ||
like peer1, peer2, peer3, etc. and we will use `null` for empty slots. | ||
|
||
This set will change over time as we will be able to add and remove members. | ||
|
||
For example, let's say we have the following set of peers: | ||
``` | ||
set = [peer1, peer2, peer3, null, null] | ||
``` | ||
|
||
Now, imagine we have a map that represent opinions from one peer to the whole group. Every opinion should match the set to be valid, in the following way: | ||
``` | ||
peer1 => [(peer1, 0), (peer2, 4), (peer3, 6), (null, 0), (null, 0)] | ||
``` | ||
The opinion array should equal the original set in length.\ | ||
The items in the array are touples of the id of the peer that we want to give the score to and the actual score. The id at each index should match the id in the set in order to be considered valid. | ||
|
||
The whole map should look like this: | ||
``` | ||
scores => { | ||
peer1 => [(peer1, 0), (peer2, 4), (peer3, 6), (null, 0), (null, 0)] | ||
peer2 => [(peer1, 4), (peer2, 0), (peer3, 6), (null, 0), (null, 0)] | ||
peer3 => [(peer1, 4), (peer2, 6), (peer3, 0), (null, 0), (null, 0)] | ||
} | ||
``` | ||
|
||
Now, let's take a look at how we filter out invalid cases from the opinion array. | ||
|
||
**Filtering of invalid cases:** | ||
|
||
1) Id at the specific index does not match the one in the set:\ | ||
Suppose we want to give a score of 5 to peer13 at the index 3: | ||
``` | ||
peer1 => [(peer1, 0), (peer2, 4), (peer3, 6), (peer13, 5), (null, 0)] | ||
``` | ||
Since the id at index 3 is null, the id and the score will be nullified, and the new opinion will look like: | ||
``` | ||
peer1 => [(peer1, 0), (peer2, 4), (peer3, 6), (null, 0), (null, 0)] | ||
``` | ||
|
||
2) Non 0 score was given to itself:\ | ||
Giving score to itself is forbiden since peers would be able to give the score only to themselves, thus introducing reputation leaking during the convergence. | ||
So, this type of opinion: | ||
``` | ||
peer1 => [(peer1, 4), (peer2, 0), (peer3, 6), (null, 0), (null, 0)] | ||
``` | ||
will turn into: | ||
``` | ||
peer1 => [(peer1, 0), (peer2, 0), (peer3, 6), (null, 0), (null, 0)] | ||
``` | ||
|
||
3) Total sum of scores is 0:\ | ||
If the initial opinion, or the filtered opinion has a sum of scores of 0, | ||
the equal score (score of 1) is given to each peer, so this: | ||
``` | ||
peer1 => [(peer1, 0), (peer2, 0), (peer3, 0), (null, 0), (null, 0)] | ||
``` | ||
will turn into this: | ||
``` | ||
peer1 => [(peer1, 0), (peer2, 1), (peer3, 1), (null, 0), (null, 0)] | ||
``` | ||
|
||
4) Opinion array does not exist/not signed:\ | ||
Will be treated the same way as 3). The equal score will be distributed to all peers | ||
|
||
The pseudo code algorithm: | ||
```rust | ||
for i in set.len() { | ||
let pk_i = set[i]; | ||
if pk_i == null { | ||
continue; | ||
} | ||
|
||
for j in set.len() { | ||
let pk_j = set[j]; | ||
let op_pk_j = scores[pk_i][j].0; | ||
|
||
let is_diff_pk_j = pk_j != op_pk_j; | ||
let is_pk_j_zero = pk_j == null; | ||
let is_pk_i = pk_j == pk_i; | ||
|
||
if is_diff_pk_j || is_pk_j_zero || is_pk_i { | ||
scores[pk_i][j].1 = 0; | ||
} | ||
|
||
if is_diff_pk_j { | ||
scores[pk_i][j].0 = pk_j; | ||
} | ||
} | ||
|
||
let op_score_sum = sum(scores[pk_i]); | ||
if op_score_sum == 0 { | ||
for j in 0..group.len() { | ||
let pk_j = scores[pk_i][j].0; | ||
|
||
let is_diff_pk = pk_j != pk_i; | ||
let is_not_null = pk_j != null; | ||
|
||
if is_diff_pk && is_not_null { | ||
scores[pk_i][j] = (pk_j, Fr::from(1)); | ||
} | ||
} | ||
} | ||
} | ||
``` |
Oops, something went wrong.