diff --git a/src/agent/services/exporter.rs b/src/agent/services/exporter.rs index 559eaa9..ac8832e 100644 --- a/src/agent/services/exporter.rs +++ b/src/agent/services/exporter.rs @@ -261,6 +261,7 @@ mod exporter { config.exporter.staleness_threshold, config.exporter.unchanged_publish_threshold, ).await { + let publisher_buffer_key = Exporter::get_publisher_buffer_key(&*state).await; if let Err(err) = publish_batches( state.clone(), client.clone(), @@ -270,7 +271,7 @@ mod exporter { &publish_keypair, key_store.oracle_program_key, key_store.publish_program_key, - key_store.publisher_buffer_key, + publisher_buffer_key, config.exporter.max_batch_size, config.exporter.staleness_threshold, config.exporter.compute_unit_limit, diff --git a/src/agent/services/oracle.rs b/src/agent/services/oracle.rs index 3876d30..7de2194 100644 --- a/src/agent/services/oracle.rs +++ b/src/agent/services/oracle.rs @@ -62,6 +62,7 @@ where state.clone(), key_store.mapping_key, key_store.publish_keypair, + key_store.publish_program_key, config.oracle.max_lookup_batch_size, ))); @@ -159,6 +160,7 @@ async fn poller( state: Arc, mapping_key: Pubkey, publish_keypair: Option, + publish_program_key: Option, max_lookup_batch_size: usize, ) where S: Oracle, @@ -183,6 +185,7 @@ async fn poller( network, mapping_key, publish_keypair.as_ref(), + publish_program_key, &client, max_lookup_batch_size, ) diff --git a/src/agent/solana.rs b/src/agent/solana.rs index ebbf4c3..3ea5858 100644 --- a/src/agent/solana.rs +++ b/src/agent/solana.rs @@ -103,13 +103,6 @@ pub mod key_store { default )] pub publish_program_key: Option, - /// The public key of the publisher's buffer for the Publish program - #[serde( - serialize_with = "opt_pubkey_string_ser", - deserialize_with = "opt_pubkey_string_de", - default - )] - pub publisher_buffer_key: Option, /// The public key of the root mapping account #[serde( serialize_with = "pubkey_string_ser", @@ -129,18 +122,15 @@ pub mod key_store { /// The keypair used to publish price updates. When None, /// publishing will not start until a new keypair is supplied /// via the remote loading endpoint - pub publish_keypair: Option, + pub publish_keypair: Option, /// Public key of the Oracle program - pub oracle_program_key: Pubkey, + pub oracle_program_key: Pubkey, /// Public key of the Publish program - pub publish_program_key: Option, - /// Public key of the publisher's buffer for the publish program - pub publisher_buffer_key: Option, - + pub publish_program_key: Option, /// Public key of the root mapping account - pub mapping_key: Pubkey, + pub mapping_key: Pubkey, /// Public key of the accumulator program (if provided) - pub accumulator_key: Option, + pub accumulator_key: Option, } impl KeyStore { @@ -161,7 +151,6 @@ pub mod key_store { publish_keypair, oracle_program_key: config.oracle_program_key, publish_program_key: config.publish_program_key, - publisher_buffer_key: config.publisher_buffer_key, mapping_key: config.mapping_key, accumulator_key: config.accumulator_key, }) diff --git a/src/agent/state/exporter.rs b/src/agent/state/exporter.rs index 514e9f6..05a9673 100644 --- a/src/agent/state/exporter.rs +++ b/src/agent/state/exporter.rs @@ -15,10 +15,16 @@ use { }, }, anyhow::{ - anyhow, bail, Context, Result + anyhow, + bail, + Context, + Result, }, bincode::Options, - bytemuck::{bytes_of, cast_slice}, + bytemuck::{ + bytes_of, + cast_slice, + }, chrono::Utc, futures_util::future::join_all, pyth_price_publisher::accounts::buffer::BufferedPrice, @@ -81,6 +87,8 @@ pub struct ExporterState { /// Currently known permissioned prices of this publisher along with their market hours our_prices: RwLock>, + publisher_buffer_key: RwLock>, + /// Recent compute unit price in micro lamports (set if dynamic compute unit pricing is enabled) recent_compute_unit_price_micro_lamports: RwLock>, } @@ -106,6 +114,7 @@ where staleness_threshold: Duration, unchanged_publish_threshold: Duration, ) -> Result>; + async fn get_publisher_buffer_key(&self) -> Option; async fn get_recent_compute_unit_price_micro_lamports(&self) -> Option; async fn update_recent_compute_unit_price( &self, @@ -114,11 +123,12 @@ where staleness_threshold: Duration, unchanged_publish_threshold: Duration, ) -> Result<()>; - async fn update_permissions( + async fn update_on_chain_state( &self, network: Network, publish_keypair: Option<&Keypair>, publisher_permissions: HashMap>, + publisher_buffer_key: Option, ) -> Result<()>; } @@ -267,6 +277,10 @@ where .collect::>()) } + async fn get_publisher_buffer_key(&self) -> Option { + *self.into().publisher_buffer_key.read().await + } + async fn get_recent_compute_unit_price_micro_lamports(&self) -> Option { *self .into() @@ -313,11 +327,12 @@ where } #[instrument(skip(self, publish_keypair, publisher_permissions))] - async fn update_permissions( + async fn update_on_chain_state( &self, network: Network, publish_keypair: Option<&Keypair>, publisher_permissions: HashMap>, + publisher_buffer_key: Option, ) -> Result<()> { let publish_keypair = get_publish_keypair(self, network, publish_keypair).await?; *self.into().our_prices.write().await = publisher_permissions @@ -330,6 +345,7 @@ where ); HashMap::new() }); + *self.into().publisher_buffer_key.write().await = publisher_buffer_key; Ok(()) } @@ -580,7 +596,8 @@ where let instruction = create_instruction_with_publish_program( publish_keypair.pubkey(), publish_program_key, - publisher_buffer_key.context("must specify publisher_buffer_key if publish_program_key is specified")?, + publisher_buffer_key + .context("must specify publisher_buffer_key if publish_program_key is specified")?, updates, )?; instructions.push(instruction); @@ -805,7 +822,11 @@ fn create_instruction_with_publish_program( publisher_buffer_key: Pubkey, prices: Vec, ) -> Result { - use pyth_price_publisher::instruction::{Instruction as PublishInstruction, SubmitPricesArgsHeader, PUBLISHER_CONFIG_SEED}; + use pyth_price_publisher::instruction::{ + Instruction as PublishInstruction, + SubmitPricesArgsHeader, + PUBLISHER_CONFIG_SEED, + }; let (publisher_config_key, publisher_config_bump) = Pubkey::find_program_address( &[PUBLISHER_CONFIG_SEED.as_bytes(), &publish_pubkey.to_bytes()], &publish_program_key, diff --git a/src/agent/state/oracle.rs b/src/agent/state/oracle.rs index 39b04b4..e343658 100644 --- a/src/agent/state/oracle.rs +++ b/src/agent/state/oracle.rs @@ -18,6 +18,7 @@ use { Context, Result, }, + pyth_price_publisher::instruction::PUBLISHER_CONFIG_SEED, pyth_sdk_solana::state::{ load_mapping_account, load_product_account, @@ -37,6 +38,7 @@ use { commitment_config::CommitmentLevel, pubkey::Pubkey, signature::Keypair, + signer::Signer, }, std::{ collections::{ @@ -135,6 +137,7 @@ pub struct Data { pub price_accounts: HashMap, /// publisher => {their permissioned price accounts => price publishing metadata} pub publisher_permissions: HashMap>, + pub publisher_buffer_key: Option, } #[derive(Clone, Serialize, Deserialize, Debug)] @@ -193,6 +196,7 @@ pub trait Oracle { network: Network, mapping_key: Pubkey, publish_keypair: Option<&Keypair>, + publish_program_key: Option, rpc_client: &RpcClient, max_lookup_batch_size: usize, ) -> Result<()>; @@ -267,6 +271,7 @@ where network: Network, mapping_key: Pubkey, publish_keypair: Option<&Keypair>, + publish_program_key: Option, rpc_client: &RpcClient, max_lookup_batch_size: usize, ) -> Result<()> { @@ -311,22 +316,44 @@ where } } + let mut publisher_buffer_key = None; + if let (Some(publish_program_key), Some(publish_keypair)) = + (publish_program_key, publish_keypair) + { + match fetch_publisher_buffer_key( + rpc_client, + publish_program_key, + publish_keypair.pubkey(), + ) + .await + { + Ok(r) => { + publisher_buffer_key = Some(r); + } + Err(err) => { + tracing::warn!("failed to fetch publisher buffer key: {:?}", err); + } + } + } + let new_data = Data { mapping_accounts, product_accounts, price_accounts, publisher_permissions, + publisher_buffer_key, }; let mut data = self.into().data.write().await; log_data_diff(&data, &new_data); *data = new_data; - Exporter::update_permissions( + Exporter::update_on_chain_state( self, network, publish_keypair, data.publisher_permissions.clone(), + data.publisher_buffer_key, ) .await?; @@ -367,6 +394,23 @@ where } } +async fn fetch_publisher_buffer_key( + rpc_client: &RpcClient, + publish_program_key: Pubkey, + publisher_pubkey: Pubkey, +) -> Result { + let (publisher_config_key, _bump) = Pubkey::find_program_address( + &[ + PUBLISHER_CONFIG_SEED.as_bytes(), + &publisher_pubkey.to_bytes(), + ], + &publish_program_key, + ); + let data = rpc_client.get_account_data(&publisher_config_key).await?; + let config = pyth_price_publisher::accounts::publisher_config::read(&data)?; + Ok(config.buffer_account.into()) +} + #[instrument(skip(rpc_client))] async fn fetch_mapping_accounts( rpc_client: &RpcClient,