Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(oracle): WIP add scores account #407

Open
wants to merge 39 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 27 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
b343951
feat(oracle): WIP add scores account
keyvankhademi Jun 7, 2024
1561e80
feat: use bitset to reduce scores account size
keyvankhademi Jun 10, 2024
b58625c
fix: pre-commit
keyvankhademi Jun 10, 2024
18c6787
fix: add price test
keyvankhademi Jun 13, 2024
32c577e
fix: rename score to cap
keyvankhademi Jun 13, 2024
2ec0004
feat: remove publisher sorting features
keyvankhademi Jun 14, 2024
a02a10b
feat: improve caps account calculations
keyvankhademi Jun 14, 2024
0c98527
feat: rename score to publisher_caps
keyvankhademi Jun 14, 2024
87f61ec
fix: bypass false positive clippy
keyvankhademi Jun 14, 2024
2f6f359
feat: optimize oracle command enum
keyvankhademi Jun 14, 2024
2697cb2
feat: test new way of calculating score
keyvankhademi Jun 14, 2024
d07a256
fix: correct rust doc format
keyvankhademi Jun 17, 2024
98a6a52
fix: clippy
keyvankhademi Jun 17, 2024
4260e2e
feat: add vscode settings
keyvankhademi Jun 17, 2024
6c9d3e6
fix: clippy warning
keyvankhademi Jun 17, 2024
8e00103
feat: change OracleCommand repr from i32 to u32
keyvankhademi Jun 17, 2024
aa1e0a7
feat: add initial tests for publisher caps
keyvankhademi Jun 17, 2024
2372be6
feat: add migration step
keyvankhademi Jun 17, 2024
63352ec
test: try a different implementation for program size
keyvankhademi Jun 17, 2024
aaa12bf
test: try manual swap to decrease program size
keyvankhademi Jun 17, 2024
e2aa645
fix: add price when initializing the caps account
keyvankhademi Jun 17, 2024
2169e8d
fix: rename scores account to cap account
keyvankhademi Jun 17, 2024
de0f7c3
refactor: rename calculate_scores to calculate_caps
keyvankhademi Jun 17, 2024
4396054
refactor: rename symbols to prices in PublisherCapsAccount
keyvankhademi Jun 17, 2024
0563db6
refactor: update publisher permission index to use price instead of s…
keyvankhademi Jun 17, 2024
6c321bf
refactor: update publisher permission index to use price instead of s…
keyvankhademi Jun 17, 2024
9e90897
chore: Update num-traits dependency to version 0.2
keyvankhademi Jun 17, 2024
5c1f750
feat: add initial quickcheck for caps account
keyvankhademi Jun 18, 2024
02f6257
refactor: update upd_price.rs to use send_message_to_message_buffer f…
keyvankhademi Jun 18, 2024
7800412
feat: send caps to message buffer for add publisher
keyvankhademi Jun 19, 2024
9b52a40
fix: comment migration code to test program size
keyvankhademi Jun 19, 2024
a18ec0c
feat: remove init mapping to reduce program size
keyvankhademi Jun 19, 2024
5932402
feat: keep init mapping for tests
keyvankhademi Jun 19, 2024
21c38c8
fix: self ref error
keyvankhademi Jun 19, 2024
413abc3
fix: revert mapping account changes
keyvankhademi Jun 19, 2024
23a1da8
feat: add cfg test flag for init mapping
keyvankhademi Jun 19, 2024
1a11ca4
temporary remove pipeline test
keyvankhademi Jun 19, 2024
1fa4c6a
remove upd_permissions to check size
keyvankhademi Jun 19, 2024
dd453fb
add feature test to be able to compile oracle for test
keyvankhademi Jun 19, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 9 additions & 0 deletions .vscode/settings.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
{
"rust-analyzer.check.command": "clippy",
"editor.defaultFormatter": "rust-lang.rust-analyzer",
"rust-analyzer.rustfmt.overrideCommand": [
"rustfmt",
"+nightly"
],
"editor.formatOnSave": true,
}
17 changes: 11 additions & 6 deletions program/c/src/oracle/oracle.h
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,12 @@ extern "C" {
#define PC_PUBKEY_SIZE 32
#define PC_PUBKEY_SIZE_64 (PC_PUBKEY_SIZE/sizeof(uint64_t))
#define PC_MAP_TABLE_SIZE 640
#define PC_MAX_PUBLISHERS 256
#define PC_MAX_SYMBOLS 1024
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what we do if we exceed this numbers?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That sounds like a Pythnet v2 problem anyway

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The program doesn't allow exceeding these numbers, we will probably face a lot of issues if we decide to increase these numbers. We are hoping to go to pythnet V2 before we need to increase these numbers. From the trasactions limit that we currently have, i don't think this will an issue.
But if you think there is a chance that we exceed these number before pythnet v2, we better address it now.

// This is the number of 64-bit words needed to store the bitset of symbols
#define PC_MAX_SYMBOLS_64 (PC_MAX_SYMBOLS/64)

// Total price component slots available
// Total price component slots available
#define PC_NUM_COMP_PYTHNET 128

// PC_NUM_COMP - number of price components in use
Expand All @@ -49,11 +53,12 @@ extern "C" {
#define PC_STATUS_IGNORED 4

// account types
#define PC_ACCTYPE_MAPPING 1
#define PC_ACCTYPE_PRODUCT 2
#define PC_ACCTYPE_PRICE 3
#define PC_ACCTYPE_TEST 4
#define PC_ACCTYPE_PERMISSIONS 5
#define PC_ACCTYPE_MAPPING 1
#define PC_ACCTYPE_PRODUCT 2
#define PC_ACCTYPE_PRICE 3
#define PC_ACCTYPE_TEST 4
#define PC_ACCTYPE_PERMISSIONS 5
#define PC_ACCTYPE_SCORE 6


// Compute budget requested per price update instruction
Expand Down
2 changes: 1 addition & 1 deletion program/rust/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@ solana-program = "=1.13.3"
bytemuck = "1.11.0"
thiserror = "1.0"
num-derive = "0.3"
num-traits = "0.2"
byteorder = "1.4.3"
serde = { version = "1.0", features = ["derive"], optional = true }
strum = { version = "0.24.1", features = ["derive"], optional = true }
Expand All @@ -26,6 +25,7 @@ tokio = "1.14.1"
hex = "0.3.1"
quickcheck = "1"
rand = "0.8.5"
num-traits = "0.2"
quickcheck_macros = "1"
bincode = "1.3.3"
serde = { version = "1.0", features = ["derive"] }
Expand Down
6 changes: 3 additions & 3 deletions program/rust/build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -89,10 +89,10 @@ fn do_make_build(targets: Vec<&str>, out_dir: &Path) {
if !make_output.status.success() {
panic!(
"C oracle make build did not exit with 0 (code
({:?}).\n\nstdout:\n{}\n\nstderr:\n{}",
({:?}).\n\nstdout:\n{}\n\nstderr:\n{}",
make_output.status.code(),
String::from_utf8(make_output.stdout).unwrap_or("<non-utf8>".to_owned()),
String::from_utf8(make_output.stderr).unwrap_or("<non-utf8>".to_owned())
String::from_utf8(make_output.stdout).unwrap_or_else(|_| "<non-utf8>".to_owned()),
String::from_utf8(make_output.stderr).unwrap_or_else(|_| "<non-utf8>".to_owned())
);
}
}
Expand Down
2 changes: 2 additions & 0 deletions program/rust/src/accounts.rs
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ mod mapping;
mod permission;
mod price;
mod product;
mod publisher_caps;

// Some types only exist during use as a library.
#[cfg(feature = "strum")]
Expand All @@ -64,6 +65,7 @@ pub use {
update_product_metadata,
ProductAccount,
},
publisher_caps::PublisherCapsAccount,
};

// PDA seeds for accounts.
Expand Down
307 changes: 307 additions & 0 deletions program/rust/src/accounts/publisher_caps.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,307 @@
use {
super::{
AccountHeader,
PythAccount,
},
crate::c_oracle_header::{
PC_ACCTYPE_SCORE,
PC_MAX_PUBLISHERS,
PC_MAX_SYMBOLS,
PC_MAX_SYMBOLS_64,
},
bytemuck::{
Pod,
Zeroable,
},
solana_program::{
program_error::ProgramError,
pubkey::Pubkey,
},
std::cmp::max,
};

#[repr(C)]
#[derive(Copy, Clone, Pod, Zeroable)]

/// This account is part of Community Integrity Pool (CIP) project.
/// It is used to store the caps of the publishers which will be sent
/// to the `integrity_pool` program on mainnet to calculate the rewards.
pub struct PublisherCapsAccount {
pub header: AccountHeader,
pub num_publishers: usize,
pub num_symbols: usize,
// Z is a constant used to normalize the caps
pub z: u64,
// M is a constant showing the target stake per symbol
pub m: u64,

// array[x][y] is a u64 whose bits represent if publisher x publishes symbols 64*y to 64*(y+1) - 1
pub publisher_permissions: [[u64; PC_MAX_SYMBOLS_64 as usize]; PC_MAX_PUBLISHERS as usize],
pub caps: [u64; PC_MAX_PUBLISHERS as usize],
pub publishers: [Pubkey; PC_MAX_PUBLISHERS as usize],
pub prices: [Pubkey; PC_MAX_SYMBOLS as usize],
}

impl PublisherCapsAccount {
fn get_publisher_permission(&self, x: usize, y: usize) -> bool {
(self.publisher_permissions[x][y / 64] >> (y % 64)) & 1 == 1
}

fn set_publisher_permission(&mut self, x: usize, y: usize, value: bool) {
if value {
self.publisher_permissions[x][y / 64] |= 1 << (y % 64);
} else {
self.publisher_permissions[x][y / 64] &= !(1 << (y % 64));
}
}

pub fn add_publisher(
&mut self,
publisher: Pubkey,
price_account: Pubkey,
) -> Result<(), ProgramError> {
let publisher_index = self
.publishers
.iter()
.position(|&x| x == publisher)
.unwrap_or(self.num_publishers);

if publisher_index == self.num_publishers {
if self.num_publishers == PC_MAX_PUBLISHERS as usize {
return Err(ProgramError::AccountDataTooSmall);
}

self.publishers[self.num_publishers] = publisher;
self.num_publishers += 1;
}

let symbol_index = self
.prices
.iter()
.position(|&x| x == price_account)
.ok_or(ProgramError::InvalidArgument)?;

self.set_publisher_permission(publisher_index, symbol_index, true);

self.calculate_caps()
}

pub fn del_publisher(
&mut self,
publisher: Pubkey,
price_account: Pubkey,
) -> Result<(), ProgramError> {
let publisher_index = self
.publishers
.iter()
.position(|&x| x == publisher)
.ok_or(ProgramError::InvalidArgument)?;

let price_index = self
.prices
.iter()
.position(|&x| x == price_account)
.ok_or(ProgramError::InvalidArgument)?;

self.set_publisher_permission(publisher_index, price_index, false);
self.calculate_caps()
}

pub fn add_price(&mut self, symbol: Pubkey) -> Result<(), ProgramError> {
let price_index = self
.prices
.iter()
.position(|&x| x == symbol)
.unwrap_or(self.num_symbols);

if price_index == self.num_symbols {
if self.num_symbols == PC_MAX_SYMBOLS as usize {
return Err(ProgramError::AccountDataTooSmall);
}

self.prices[self.num_symbols] = symbol;
self.num_symbols += 1;
}

Ok(())
}

pub fn del_price(&mut self, symbol: Pubkey) -> Result<(), ProgramError> {
let price_index = self
.prices
.iter()
.position(|&x| x == symbol)
.ok_or(ProgramError::InvalidArgument)?;

// update symbol list
self.prices[price_index] = self.prices[self.num_symbols - 1];
self.prices[self.num_symbols - 1] = Pubkey::default();

// update publisher permissions
for i in 0..self.num_publishers {
let value = self.get_publisher_permission(i, self.num_symbols - 1);
self.set_publisher_permission(i, price_index, value)
}

self.num_symbols -= 1;
self.calculate_caps()
}


pub fn calculate_caps(&mut self) -> Result<(), ProgramError> {
let price_weights: Vec<u64> = self
.prices
.iter()
.enumerate()
.map(|(j, _)| {
let weight = self
.publisher_permissions
.iter()
.enumerate()
.filter(|(i, _)| self.get_publisher_permission(*i, j))
.count() as u64;
max(weight, self.z)
})
.collect();

for i in 0..self.num_publishers {
self.caps[i] = self
.prices
.iter()
.enumerate()
.filter(|(j, _)| self.get_publisher_permission(i, *j))
.map(|(j, _)| self.m * 1_000_000_000_u64 / price_weights[j])
.sum();
}
Ok(())
}
}

impl PythAccount for PublisherCapsAccount {
const ACCOUNT_TYPE: u32 = PC_ACCTYPE_SCORE;
// Calculate the initial size of the account
const INITIAL_SIZE: u32 = 75824;
}


#[cfg(test)]
mod tests {
use {
super::*,
crate::c_oracle_header::{
PC_ACCTYPE_SCORE,
PC_VERSION,
},
solana_program::pubkey::Pubkey,
std::mem::size_of,
};

fn get_empty_account() -> PublisherCapsAccount {
PublisherCapsAccount {
header: AccountHeader {
magic_number: PC_ACCTYPE_SCORE,
account_type: PC_ACCTYPE_SCORE,
version: PC_VERSION,
size: size_of::<PublisherCapsAccount>() as u32,
},
num_publishers: 0,
num_symbols: 0,
z: 2,
m: 1000,
publisher_permissions: [[0; PC_MAX_SYMBOLS_64 as usize]; PC_MAX_PUBLISHERS as usize],
caps: [0; PC_MAX_PUBLISHERS as usize],
publishers: [Pubkey::default(); PC_MAX_PUBLISHERS as usize],
prices: [Pubkey::default(); PC_MAX_SYMBOLS as usize],
}
}

#[test]
fn test_size_of_publisher_caps_account() {
let account = get_empty_account();
assert_eq!(
PublisherCapsAccount::INITIAL_SIZE as usize,
size_of::<PublisherCapsAccount>()
);
assert_eq!(
account.header.size as usize,
size_of::<PublisherCapsAccount>()
);
}

#[test]
fn test_publisher_caps_account() {
let mut account = get_empty_account();

let publisher1 = Pubkey::new_unique();
let publisher2 = Pubkey::new_unique();
let publisher3 = Pubkey::new_unique();
let symbol1 = Pubkey::new_unique();
let symbol2 = Pubkey::new_unique();

account.add_price(symbol1).unwrap();
account.add_price(symbol2).unwrap();

account.add_publisher(publisher1, symbol1).unwrap();
assert_eq!(account.caps[..2], [account.m * 500_000_000, 0]);

account.add_publisher(publisher1, symbol2).unwrap();
assert!(account.get_publisher_permission(0, 0));
assert!(account.get_publisher_permission(0, 1));
assert_eq!(account.caps[..2], [account.m * 1_000_000_000, 0]);


account.add_publisher(publisher2, symbol1).unwrap();
assert_eq!(
account.caps[..2],
[account.m * 1_000_000_000, account.m * 500_000_000]
);

account.add_publisher(publisher2, symbol2).unwrap();
assert!(account.get_publisher_permission(1, 0));
assert!(account.get_publisher_permission(1, 1));
assert_eq!(
account.caps[..2],
[account.m * 1_000_000_000, account.m * 1_000_000_000]
);

account.add_publisher(publisher3, symbol1).unwrap();
assert_eq!(
account.caps[..3],
[833_333_333_333, 833_333_333_333, 333333333333]
);

account.del_publisher(publisher1, symbol1).unwrap();
assert_eq!(account.num_publishers, 3);
assert_eq!(account.num_symbols, 2);
assert!(!account.get_publisher_permission(0, 0));
assert!(account.get_publisher_permission(0, 1));
assert!(account.get_publisher_permission(1, 0));
assert!(account.get_publisher_permission(1, 1));

account.del_publisher(publisher1, symbol2).unwrap();
assert_eq!(account.num_publishers, 3);
assert_eq!(account.num_symbols, 2);
assert!(!account.get_publisher_permission(0, 0));
assert!(!account.get_publisher_permission(0, 1));
assert!(account.get_publisher_permission(1, 0));
assert!(account.get_publisher_permission(1, 1));

account.del_publisher(publisher2, symbol1).unwrap();
assert_eq!(account.num_publishers, 3);
assert_eq!(account.num_symbols, 2);
assert!(!account.get_publisher_permission(0, 0));
assert!(!account.get_publisher_permission(0, 1));
assert!(!account.get_publisher_permission(1, 0));
assert!(account.get_publisher_permission(1, 1));

account.del_publisher(publisher2, symbol2).unwrap();
assert_eq!(account.num_publishers, 3);
assert_eq!(account.num_symbols, 2);
assert!(!account.get_publisher_permission(0, 0));
assert!(!account.get_publisher_permission(0, 1));
assert!(!account.get_publisher_permission(1, 0));
assert!(!account.get_publisher_permission(1, 1));

assert_eq!(account.caps[..3], [0, 0, 500000000000]);
}
}
Loading
Loading