feat(target_chains/starknet): fee collection (#1527)

* feat(target_chains/starknet): fee collection

* refactor(target_chains/starknet): renames and comments
This commit is contained in:
Pavel Strakhov 2024-04-30 23:03:29 +01:00 committed by GitHub
parent 20d99bceb7
commit 4e630edac0
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 78 additions and 4 deletions

View File

@ -1,10 +1,16 @@
# Code generated by scarb DO NOT EDIT.
version = 1
[[package]]
name = "openzeppelin"
version = "0.10.0"
source = "git+https://github.com/OpenZeppelin/cairo-contracts.git?tag=v0.10.0#d77082732daab2690ba50742ea41080eb23299d3"
[[package]]
name = "pyth"
version = "0.1.0"
dependencies = [
"openzeppelin",
"snforge_std",
]

View File

@ -5,6 +5,8 @@ edition = "2023_11"
[dependencies]
starknet = ">=2.5.4"
openzeppelin = { git = "https://github.com/OpenZeppelin/cairo-contracts.git", tag = "v0.10.0" }
snforge_std = { git = "https://github.com/foundry-rs/starknet-foundry", tag = "v0.21.0" }
[[target.starknet-contract]]
build-external-contracts = ["openzeppelin::presets::erc20::ERC20"]

View File

@ -22,7 +22,8 @@ wormhole_hash=$(starkli declare target/dev/pyth_wormhole.contract_class.json)
# prefunded katana account
owner=0x6162896d1d7ab204c7ccac6dd5f8e9e7c25ecd5ae4fcb4ad32e57786bb46e03
fee_token_address=0x49d36570d4e46f48e99674bd3fcc84644ddd6b96f7c741b1562b82f9e004dc7
# predeployed fee token contract in katana
fee_contract_address=0x49d36570d4e46f48e99674bd3fcc84644ddd6b96f7c741b1562b82f9e004dc7
# deploying wormhole with mainnet guardians
@ -119,7 +120,7 @@ ${sleep}
pyth_address=$(starkli deploy "${pyth_hash}" \
"${owner}" \
"${wormhole_address}" \
"${fee_token_address}" \
"${fee_contract_address}" \
1000 0 `# fee amount` \
1 `# num_data_sources` \
26 `# emitter_chain_id` \
@ -127,7 +128,10 @@ pyth_address=$(starkli deploy "${pyth_hash}" \
)
${sleep}
starkli invoke "${pyth_address}" update_price_feeds \
starkli invoke "${fee_contract_address}" approve "${pyth_address}" 1000 0
${sleep}
starkli invoke --watch --log-traffic "${pyth_address}" update_price_feeds \
11 41 141887862745809943100717722154781668316147089807066324001213790862261653767 451230040559159019530944948086670994623010697390864133264612902902585665886 355897384610106978643111834734000274494997301794613218547634257521495150151 140511063638834349363702006999356227863549404051701803148734324248522745879 435849190784772134907557391544163070978531038970298390345939133663347953446 416390591179833928094641114955594939466104495718036761707729297119441316151 360454929416220920336539568461651500076647166763464050800345920693176904002 316054999864337699543932294956493808847640383114707243342262764542081441331 325277902980160684959962429721294603784343718796390808940252812862355246813 43683235854839458868457367619068018785880460427473556950900276498953667 448289429405712011882317781416869052550573589492688760675666957663813001522 118081463902430977133121147164253483958565039026724621562859841189218059803 194064310618695309465615383754562031677972810736048112738513050109934134235 133901765334590923121691219814784557892214901646312752962904032795881821509 404227501001709279944936006741063968912686453006275462577777397594240621266 81649001731335394114026683805238949464016657447685509824621946636993704965 32402065226491532148674904435794801976788068837745943243341272676331333141 431262841416902409381606630149292665102873776020834630861578112749151562174 6164523115980545628843981978797257048781800754033825701059814297149591186 408761574582108996678203805090470134287794603493622537384530614829262728153 185368533577943244707350150853170361880334596276529206938783888784867529821 173578821500714074579643724957224629379984215847383417303110192934676518530 90209855380378362490166376523380463998928070428866100240907090599465187835 97758466908511588082569287391708453107999243934457382895073183209581711489 132725011490528489913736834798247512772139171145730373610858422315799224432 117123868005849140967825260063167768530251411611975150066586827543934313288 408149062252618928234854115279677715692278734600386004492580987016428761675 164529520317122600276020522906605877985809506451193373524142111430138855019 444793051809958482843529748761971363435331354795896511243191618771787268378 247660009137502548346315865368477795392972486141407800140910365553760622080 3281582060272565111592312037403686940429019548922889497694300188 93649805131515836129946966966350066506512123780266587069413066350925286142 394112423559676785086098106350541172262729583743734966358666094809121292390 35403101004688876764673991514113473446030702766599795822870037077688984558 99366103604611980443183454746643823071419076016677225828619807954313149423 10381657217606191031071521950784155484751645280452344547752823767622424055 391045354044274401116419632681482293741435113770205621235865697077178955228 311250087759201408758984550959714865999349469611700431708031036894849650573 59953730895385399344628932835545900304309851622811198425230584225200786697 226866843267230707879834616967256711063296411939069440476882347301771901839 95752383404870925303422787
echo Pyth contract has been successfully deployed at "${pyth_address}"

View File

@ -66,6 +66,7 @@ pub enum UpdatePriceFeedsError {
Wormhole: super::wormhole::ParseAndVerifyVmError,
InvalidUpdateData,
InvalidUpdateDataSource,
InsufficientFeeAllowance,
}
pub impl UpdatePriceFeedsErrorUnwrapWithFelt252<T> of UnwrapWithFelt252<T, UpdatePriceFeedsError> {
@ -84,6 +85,7 @@ impl UpdatePriceFeedsErrorIntoFelt252 of Into<UpdatePriceFeedsError, felt252> {
UpdatePriceFeedsError::Wormhole(err) => err.into(),
UpdatePriceFeedsError::InvalidUpdateData => 'invalid update data',
UpdatePriceFeedsError::InvalidUpdateDataSource => 'invalid update data source',
UpdatePriceFeedsError::InsufficientFeeAllowance => 'insufficient fee allowance',
}
}
}
@ -128,6 +130,7 @@ mod pyth {
use pyth::hash::{Hasher, HasherImpl};
use core::fmt::{Debug, Formatter};
use pyth::util::{u64_as_i64, u32_as_i32};
use openzeppelin::token::erc20::interface::{IERC20CamelDispatcherTrait, IERC20CamelDispatcher};
// Stands for PNAU (Pyth Network Accumulator Update)
const ACCUMULATOR_MAGIC: u32 = 0x504e4155;
@ -232,6 +235,22 @@ mod pyth {
latest_price_info: LegacyMap<u256, PriceInfo>,
}
/// Initializes the Pyth contract.
///
/// `owner` is the address that will be allowed to call governance methods (it's a placeholder
/// until we implement governance properly).
///
/// `wormhole_address` is the address of the deployed Wormhole contract implemented in the `wormhole` module.
///
/// `fee_contract_address` is the address of the ERC20 token used to pay fees to Pyth
/// for price updates. There is no native token on Starknet so an ERC20 contract has to be used.
/// On Katana, an ETH fee contract is pre-deployed. On Starknet testnet, ETH and STRK fee tokens are
/// available. Any other ERC20-compatible token can also be used.
/// In a Starknet Forge testing environment, a fee contract must be deployed manually.
///
/// `single_update_fee` is the number of tokens of `fee_contract_address` charged for a single price update.
///
/// `data_sources` is the list of Wormhole data sources accepted by this contract.
#[constructor]
fn constructor(
ref self: ContractState,
@ -392,6 +411,20 @@ mod pyth {
let num_updates = reader.read_u8().map_err()?;
let total_fee = get_total_fee(ref self, num_updates);
let fee_contract = IERC20CamelDispatcher {
contract_address: self.fee_contract_address.read()
};
let execution_info = get_execution_info().unbox();
let caller = execution_info.caller_address;
let contract = execution_info.contract_address;
if fee_contract.allowance(caller, contract) < total_fee {
return Result::Err(UpdatePriceFeedsError::InsufficientFeeAllowance);
}
if !fee_contract.transferFrom(caller, contract, total_fee) {
return Result::Err(UpdatePriceFeedsError::InsufficientFeeAllowance);
}
let mut i = 0;
let mut result = Result::Ok(());
while i < num_updates {
@ -468,4 +501,8 @@ mod pyth {
self.emit(event);
}
}
fn get_total_fee(ref self: ContractState, num_updates: u8) -> u256 {
self.single_update_fee.read() * num_updates.into()
}
}

View File

@ -8,6 +8,7 @@ use pyth::pyth::{
use pyth::byte_array::{ByteArray, ByteArrayImpl};
use pyth::util::{array_felt252_to_bytes31, UnwrapWithFelt252};
use core::starknet::ContractAddress;
use openzeppelin::token::erc20::interface::{IERC20CamelDispatcher, IERC20CamelDispatcherTrait};
fn decode_event(event: @Event) -> PythEvent {
if *event.keys.at(0) == event_name_hash('PriceFeedUpdate') {
@ -31,11 +32,13 @@ fn decode_event(event: @Event) -> PythEvent {
#[test]
fn update_price_feeds_works() {
let owner = 'owner'.try_into().unwrap();
let user = 'user'.try_into().unwrap();
let wormhole = super::wormhole::deploy_and_init(owner);
let fee_contract = deploy_fee_contract(user);
let pyth = deploy(
owner,
wormhole.contract_address,
0x42.try_into().unwrap(),
fee_contract.contract_address,
1000,
array![
DataSource {
@ -45,9 +48,15 @@ fn update_price_feeds_works() {
]
);
start_prank(CheatTarget::One(fee_contract.contract_address), user.try_into().unwrap());
fee_contract.approve(pyth.contract_address, 10000);
stop_prank(CheatTarget::One(fee_contract.contract_address));
let mut spy = spy_events(SpyOn::One(pyth.contract_address));
start_prank(CheatTarget::One(pyth.contract_address), user.try_into().unwrap());
pyth.update_price_feeds(good_update1()).unwrap_with_felt252();
stop_prank(CheatTarget::One(pyth.contract_address));
spy.fetch_events();
assert!(spy.events.len() == 1);
@ -100,6 +109,22 @@ fn deploy(
IPythDispatcher { contract_address }
}
fn deploy_fee_contract(recipient: ContractAddress) -> IERC20CamelDispatcher {
let mut args = array![];
let name: core::byte_array::ByteArray = "eth";
let symbol: core::byte_array::ByteArray = "eth";
(name, symbol, 100000_u256, recipient).serialize(ref args);
let contract = declare("ERC20");
let contract_address = match contract.deploy(@args) {
Result::Ok(v) => { v },
Result::Err(err) => {
panic(err.panic_data);
0.try_into().unwrap()
},
};
IERC20CamelDispatcher { contract_address }
}
// A random update pulled from Hermes.
fn good_update1() -> ByteArray {
let bytes = array![