Reorg Storage program to look more like the others
This commit is contained in:
		| @@ -22,7 +22,7 @@ use solana_client::thin_client::{create_client, ThinClient}; | |||||||
| use solana_drone::drone::{request_airdrop_transaction, DRONE_PORT}; | use solana_drone::drone::{request_airdrop_transaction, DRONE_PORT}; | ||||||
| use solana_sdk::hash::{Hash, Hasher}; | use solana_sdk::hash::{Hash, Hasher}; | ||||||
| use solana_sdk::signature::{Keypair, KeypairUtil, Signature}; | use solana_sdk::signature::{Keypair, KeypairUtil, Signature}; | ||||||
| use solana_storage_api::StorageTransaction; | use solana_storage_api::storage_transaction::StorageTransaction; | ||||||
| use std::fs::File; | use std::fs::File; | ||||||
| use std::io; | use std::io; | ||||||
| use std::io::BufReader; | use std::io::BufReader; | ||||||
|   | |||||||
| @@ -17,7 +17,8 @@ use solana_sdk::hash::Hash; | |||||||
| use solana_sdk::pubkey::Pubkey; | use solana_sdk::pubkey::Pubkey; | ||||||
| use solana_sdk::signature::{Keypair, Signature}; | use solana_sdk::signature::{Keypair, Signature}; | ||||||
| use solana_sdk::transaction::Transaction; | use solana_sdk::transaction::Transaction; | ||||||
| use solana_storage_api::{self, StorageProgram, StorageTransaction}; | use solana_storage_api::storage_instruction::StorageInstruction; | ||||||
|  | use solana_storage_api::storage_transaction::StorageTransaction; | ||||||
| use std::collections::HashSet; | use std::collections::HashSet; | ||||||
| use std::io; | use std::io; | ||||||
| use std::mem::size_of; | use std::mem::size_of; | ||||||
| @@ -374,7 +375,7 @@ impl StorageStage { | |||||||
|                 for (i, program_id) in tx.program_ids.iter().enumerate() { |                 for (i, program_id) in tx.program_ids.iter().enumerate() { | ||||||
|                     if solana_storage_api::check_id(&program_id) { |                     if solana_storage_api::check_id(&program_id) { | ||||||
|                         match deserialize(&tx.instructions[i].data) { |                         match deserialize(&tx.instructions[i].data) { | ||||||
|                             Ok(StorageProgram::SubmitMiningProof { |                             Ok(StorageInstruction::SubmitMiningProof { | ||||||
|                                 entry_height: proof_entry_height, |                                 entry_height: proof_entry_height, | ||||||
|                                 signature, |                                 signature, | ||||||
|                                 .. |                                 .. | ||||||
| @@ -453,23 +454,17 @@ impl Service for StorageStage { | |||||||
|  |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|  |     use super::*; | ||||||
|     use crate::blocktree::{create_new_tmp_ledger, Blocktree}; |     use crate::blocktree::{create_new_tmp_ledger, Blocktree}; | ||||||
|     use crate::cluster_info::ClusterInfo; |     use crate::cluster_info::ClusterInfo; | ||||||
|     use crate::contact_info::ContactInfo; |     use crate::contact_info::ContactInfo; | ||||||
|     use crate::entry::{make_tiny_test_entries, Entry}; |     use crate::entry::{make_tiny_test_entries, Entry}; | ||||||
|     use crate::service::Service; |     use crate::service::Service; | ||||||
|     use crate::storage_stage::StorageState; |  | ||||||
|     use crate::storage_stage::NUM_IDENTITIES; |  | ||||||
|     use crate::storage_stage::{ |  | ||||||
|         get_identity_index_from_signature, StorageStage, STORAGE_ROTATE_TEST_COUNT, |  | ||||||
|     }; |  | ||||||
|     use rayon::prelude::*; |     use rayon::prelude::*; | ||||||
|     use solana_sdk::genesis_block::GenesisBlock; |     use solana_sdk::genesis_block::GenesisBlock; | ||||||
|     use solana_sdk::hash::Hash; |     use solana_sdk::hash::{Hash, Hasher}; | ||||||
|     use solana_sdk::hash::Hasher; |  | ||||||
|     use solana_sdk::pubkey::Pubkey; |     use solana_sdk::pubkey::Pubkey; | ||||||
|     use solana_sdk::signature::{Keypair, KeypairUtil}; |     use solana_sdk::signature::{Keypair, KeypairUtil}; | ||||||
|     use solana_storage_api::StorageTransaction; |  | ||||||
|     use std::cmp::{max, min}; |     use std::cmp::{max, min}; | ||||||
|     use std::fs::remove_dir_all; |     use std::fs::remove_dir_all; | ||||||
|     use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering}; |     use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering}; | ||||||
|   | |||||||
| @@ -1,10 +1,9 @@ | |||||||
|  | pub mod storage_instruction; | ||||||
| pub mod storage_processor; | pub mod storage_processor; | ||||||
|  | pub mod storage_state; | ||||||
|  | pub mod storage_transaction; | ||||||
|  |  | ||||||
| use serde_derive::{Deserialize, Serialize}; |  | ||||||
| use solana_sdk::hash::Hash; |  | ||||||
| use solana_sdk::pubkey::Pubkey; | use solana_sdk::pubkey::Pubkey; | ||||||
| use solana_sdk::signature::{Keypair, Signature}; |  | ||||||
| use solana_sdk::transaction::Transaction; |  | ||||||
|  |  | ||||||
| pub const ENTRIES_PER_SEGMENT: u64 = 16; | pub const ENTRIES_PER_SEGMENT: u64 = 16; | ||||||
|  |  | ||||||
| @@ -12,58 +11,6 @@ pub fn get_segment_from_entry(entry_height: u64) -> usize { | |||||||
|     (entry_height / ENTRIES_PER_SEGMENT) as usize |     (entry_height / ENTRIES_PER_SEGMENT) as usize | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Serialize, Deserialize, Clone)] |  | ||||||
| pub enum ProofStatus { |  | ||||||
|     Valid, |  | ||||||
|     NotValid, |  | ||||||
|     Skipped, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Default, Debug, Serialize, Deserialize, Clone)] |  | ||||||
| pub struct ProofInfo { |  | ||||||
|     pub id: Pubkey, |  | ||||||
|     pub signature: Signature, |  | ||||||
|     pub sha_state: Hash, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Default, Debug, Serialize, Deserialize, Clone)] |  | ||||||
| pub struct ValidationInfo { |  | ||||||
|     pub id: Pubkey, |  | ||||||
|     pub proof_mask: Vec<ProofStatus>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Default, Debug, Serialize, Deserialize)] |  | ||||||
| pub struct StorageProgramState { |  | ||||||
|     pub entry_height: u64, |  | ||||||
|     pub hash: Hash, |  | ||||||
|  |  | ||||||
|     pub proofs: Vec<Vec<ProofInfo>>, |  | ||||||
|     pub previous_proofs: Vec<Vec<ProofInfo>>, |  | ||||||
|  |  | ||||||
|     pub lockout_validations: Vec<Vec<ValidationInfo>>, |  | ||||||
|     pub reward_validations: Vec<Vec<ValidationInfo>>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize, Debug, Clone)] |  | ||||||
| pub enum StorageProgram { |  | ||||||
|     SubmitMiningProof { |  | ||||||
|         sha_state: Hash, |  | ||||||
|         entry_height: u64, |  | ||||||
|         signature: Signature, |  | ||||||
|     }, |  | ||||||
|     AdvertiseStorageRecentBlockhash { |  | ||||||
|         hash: Hash, |  | ||||||
|         entry_height: u64, |  | ||||||
|     }, |  | ||||||
|     ClaimStorageReward { |  | ||||||
|         entry_height: u64, |  | ||||||
|     }, |  | ||||||
|     ProofValidation { |  | ||||||
|         entry_height: u64, |  | ||||||
|         proof_mask: Vec<ProofStatus>, |  | ||||||
|     }, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const STORAGE_PROGRAM_ID: [u8; 32] = [ | const STORAGE_PROGRAM_ID: [u8; 32] = [ | ||||||
|     130, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, |     130, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, | ||||||
|     0, |     0, | ||||||
| @@ -76,57 +23,3 @@ pub fn check_id(program_id: &Pubkey) -> bool { | |||||||
| pub fn id() -> Pubkey { | pub fn id() -> Pubkey { | ||||||
|     Pubkey::new(&STORAGE_PROGRAM_ID) |     Pubkey::new(&STORAGE_PROGRAM_ID) | ||||||
| } | } | ||||||
|  |  | ||||||
| pub struct StorageTransaction {} |  | ||||||
|  |  | ||||||
| impl StorageTransaction { |  | ||||||
|     pub fn new_mining_proof( |  | ||||||
|         from_keypair: &Keypair, |  | ||||||
|         sha_state: Hash, |  | ||||||
|         recent_blockhash: Hash, |  | ||||||
|         entry_height: u64, |  | ||||||
|         signature: Signature, |  | ||||||
|     ) -> Transaction { |  | ||||||
|         let program = StorageProgram::SubmitMiningProof { |  | ||||||
|             sha_state, |  | ||||||
|             entry_height, |  | ||||||
|             signature, |  | ||||||
|         }; |  | ||||||
|         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn new_advertise_recent_blockhash( |  | ||||||
|         from_keypair: &Keypair, |  | ||||||
|         storage_hash: Hash, |  | ||||||
|         recent_blockhash: Hash, |  | ||||||
|         entry_height: u64, |  | ||||||
|     ) -> Transaction { |  | ||||||
|         let program = StorageProgram::AdvertiseStorageRecentBlockhash { |  | ||||||
|             hash: storage_hash, |  | ||||||
|             entry_height, |  | ||||||
|         }; |  | ||||||
|         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn new_proof_validation( |  | ||||||
|         from_keypair: &Keypair, |  | ||||||
|         recent_blockhash: Hash, |  | ||||||
|         entry_height: u64, |  | ||||||
|         proof_mask: Vec<ProofStatus>, |  | ||||||
|     ) -> Transaction { |  | ||||||
|         let program = StorageProgram::ProofValidation { |  | ||||||
|             entry_height, |  | ||||||
|             proof_mask, |  | ||||||
|         }; |  | ||||||
|         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn new_reward_claim( |  | ||||||
|         from_keypair: &Keypair, |  | ||||||
|         recent_blockhash: Hash, |  | ||||||
|         entry_height: u64, |  | ||||||
|     ) -> Transaction { |  | ||||||
|         let program = StorageProgram::ClaimStorageReward { entry_height }; |  | ||||||
|         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								programs/storage_api/src/storage_instruction.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								programs/storage_api/src/storage_instruction.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | use crate::storage_state::ProofStatus; | ||||||
|  | use serde_derive::{Deserialize, Serialize}; | ||||||
|  | use solana_sdk::hash::Hash; | ||||||
|  | use solana_sdk::signature::Signature; | ||||||
|  |  | ||||||
|  | #[derive(Serialize, Deserialize, Debug, Clone)] | ||||||
|  | pub enum StorageInstruction { | ||||||
|  |     SubmitMiningProof { | ||||||
|  |         sha_state: Hash, | ||||||
|  |         entry_height: u64, | ||||||
|  |         signature: Signature, | ||||||
|  |     }, | ||||||
|  |     AdvertiseStorageRecentBlockhash { | ||||||
|  |         hash: Hash, | ||||||
|  |         entry_height: u64, | ||||||
|  |     }, | ||||||
|  |     ClaimStorageReward { | ||||||
|  |         entry_height: u64, | ||||||
|  |     }, | ||||||
|  |     ProofValidation { | ||||||
|  |         entry_height: u64, | ||||||
|  |         proof_mask: Vec<ProofStatus>, | ||||||
|  |     }, | ||||||
|  | } | ||||||
| @@ -2,7 +2,9 @@ | |||||||
| //!  Receive mining proofs from miners, validate the answers | //!  Receive mining proofs from miners, validate the answers | ||||||
| //!  and give reward for good proofs. | //!  and give reward for good proofs. | ||||||
|  |  | ||||||
| use crate::*; | use crate::storage_instruction::StorageInstruction; | ||||||
|  | use crate::storage_state::{ProofInfo, ProofStatus, StorageState, ValidationInfo}; | ||||||
|  | use crate::{get_segment_from_entry, ENTRIES_PER_SEGMENT}; | ||||||
| use log::*; | use log::*; | ||||||
| use solana_sdk::account::KeyedAccount; | use solana_sdk::account::KeyedAccount; | ||||||
| use solana_sdk::instruction::InstructionError; | use solana_sdk::instruction::InstructionError; | ||||||
| @@ -47,7 +49,7 @@ pub fn process_instruction( | |||||||
|         { |         { | ||||||
|             storage_account_state |             storage_account_state | ||||||
|         } else { |         } else { | ||||||
|             StorageProgramState::default() |             StorageState::default() | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         debug!( |         debug!( | ||||||
| @@ -55,7 +57,7 @@ pub fn process_instruction( | |||||||
|             storage_account_state.entry_height |             storage_account_state.entry_height | ||||||
|         ); |         ); | ||||||
|         match syscall { |         match syscall { | ||||||
|             StorageProgram::SubmitMiningProof { |             StorageInstruction::SubmitMiningProof { | ||||||
|                 sha_state, |                 sha_state, | ||||||
|                 entry_height, |                 entry_height, | ||||||
|                 signature, |                 signature, | ||||||
| @@ -79,7 +81,7 @@ pub fn process_instruction( | |||||||
|                 }; |                 }; | ||||||
|                 storage_account_state.proofs[segment_index].push(proof_info); |                 storage_account_state.proofs[segment_index].push(proof_info); | ||||||
|             } |             } | ||||||
|             StorageProgram::AdvertiseStorageRecentBlockhash { hash, entry_height } => { |             StorageInstruction::AdvertiseStorageRecentBlockhash { hash, entry_height } => { | ||||||
|                 let original_segments = storage_account_state.entry_height / ENTRIES_PER_SEGMENT; |                 let original_segments = storage_account_state.entry_height / ENTRIES_PER_SEGMENT; | ||||||
|                 let segments = entry_height / ENTRIES_PER_SEGMENT; |                 let segments = entry_height / ENTRIES_PER_SEGMENT; | ||||||
|                 debug!( |                 debug!( | ||||||
| @@ -108,7 +110,7 @@ pub fn process_instruction( | |||||||
|                     .lockout_validations |                     .lockout_validations | ||||||
|                     .resize(segments as usize, Vec::new()); |                     .resize(segments as usize, Vec::new()); | ||||||
|             } |             } | ||||||
|             StorageProgram::ProofValidation { |             StorageInstruction::ProofValidation { | ||||||
|                 entry_height, |                 entry_height, | ||||||
|                 proof_mask, |                 proof_mask, | ||||||
|             } => { |             } => { | ||||||
| @@ -134,7 +136,7 @@ pub fn process_instruction( | |||||||
|                 }; |                 }; | ||||||
|                 storage_account_state.lockout_validations[segment_index].push(info); |                 storage_account_state.lockout_validations[segment_index].push(info); | ||||||
|             } |             } | ||||||
|             StorageProgram::ClaimStorageReward { entry_height } => { |             StorageInstruction::ClaimStorageReward { entry_height } => { | ||||||
|                 let claims_index = get_segment_from_entry(entry_height); |                 let claims_index = get_segment_from_entry(entry_height); | ||||||
|                 let account_key = keyed_accounts[0].signer_key().unwrap(); |                 let account_key = keyed_accounts[0].signer_key().unwrap(); | ||||||
|                 let mut num_validations = 0; |                 let mut num_validations = 0; | ||||||
| @@ -173,7 +175,10 @@ pub fn process_instruction( | |||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|     use super::*; |     use super::*; | ||||||
|     use crate::{ProofStatus, StorageTransaction, ENTRIES_PER_SEGMENT}; |     use crate::id; | ||||||
|  |     use crate::storage_state::ProofStatus; | ||||||
|  |     use crate::storage_transaction::StorageTransaction; | ||||||
|  |     use crate::ENTRIES_PER_SEGMENT; | ||||||
|     use bincode::deserialize; |     use bincode::deserialize; | ||||||
|     use solana_runtime::bank::Bank; |     use solana_runtime::bank::Bank; | ||||||
|     use solana_sdk::account::{create_keyed_accounts, Account}; |     use solana_sdk::account::{create_keyed_accounts, Account}; | ||||||
| @@ -370,7 +375,7 @@ mod tests { | |||||||
|             Some(storage_system_account) => { |             Some(storage_system_account) => { | ||||||
|                 let state = deserialize(&storage_system_account.data); |                 let state = deserialize(&storage_system_account.data); | ||||||
|                 if let Ok(state) = state { |                 if let Ok(state) = state { | ||||||
|                     let state: StorageProgramState = state; |                     let state: StorageState = state; | ||||||
|                     return state.entry_height; |                     return state.entry_height; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -385,7 +390,7 @@ mod tests { | |||||||
|         if let Some(storage_system_account) = bank.get_account(&account) { |         if let Some(storage_system_account) = bank.get_account(&account) { | ||||||
|             let state = deserialize(&storage_system_account.data); |             let state = deserialize(&storage_system_account.data); | ||||||
|             if let Ok(state) = state { |             if let Ok(state) = state { | ||||||
|                 let state: StorageProgramState = state; |                 let state: StorageState = state; | ||||||
|                 return state.hash; |                 return state.hash; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								programs/storage_api/src/storage_state.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								programs/storage_api/src/storage_state.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | use serde_derive::{Deserialize, Serialize}; | ||||||
|  | use solana_sdk::hash::Hash; | ||||||
|  | use solana_sdk::pubkey::Pubkey; | ||||||
|  | use solana_sdk::signature::Signature; | ||||||
|  |  | ||||||
|  | #[derive(Debug, Serialize, Deserialize, Clone)] | ||||||
|  | pub enum ProofStatus { | ||||||
|  |     Valid, | ||||||
|  |     NotValid, | ||||||
|  |     Skipped, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Default, Debug, Serialize, Deserialize, Clone)] | ||||||
|  | pub struct ProofInfo { | ||||||
|  |     pub id: Pubkey, | ||||||
|  |     pub signature: Signature, | ||||||
|  |     pub sha_state: Hash, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Default, Debug, Serialize, Deserialize, Clone)] | ||||||
|  | pub struct ValidationInfo { | ||||||
|  |     pub id: Pubkey, | ||||||
|  |     pub proof_mask: Vec<ProofStatus>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Default, Debug, Serialize, Deserialize)] | ||||||
|  | pub struct StorageState { | ||||||
|  |     pub entry_height: u64, | ||||||
|  |     pub hash: Hash, | ||||||
|  |  | ||||||
|  |     pub proofs: Vec<Vec<ProofInfo>>, | ||||||
|  |     pub previous_proofs: Vec<Vec<ProofInfo>>, | ||||||
|  |  | ||||||
|  |     pub lockout_validations: Vec<Vec<ValidationInfo>>, | ||||||
|  |     pub reward_validations: Vec<Vec<ValidationInfo>>, | ||||||
|  | } | ||||||
							
								
								
									
										60
									
								
								programs/storage_api/src/storage_transaction.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								programs/storage_api/src/storage_transaction.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | use crate::id; | ||||||
|  | use crate::storage_instruction::StorageInstruction; | ||||||
|  | use crate::storage_state::ProofStatus; | ||||||
|  | use solana_sdk::hash::Hash; | ||||||
|  | use solana_sdk::signature::{Keypair, Signature}; | ||||||
|  | use solana_sdk::transaction::Transaction; | ||||||
|  |  | ||||||
|  | pub struct StorageTransaction {} | ||||||
|  |  | ||||||
|  | impl StorageTransaction { | ||||||
|  |     pub fn new_mining_proof( | ||||||
|  |         from_keypair: &Keypair, | ||||||
|  |         sha_state: Hash, | ||||||
|  |         recent_blockhash: Hash, | ||||||
|  |         entry_height: u64, | ||||||
|  |         signature: Signature, | ||||||
|  |     ) -> Transaction { | ||||||
|  |         let program = StorageInstruction::SubmitMiningProof { | ||||||
|  |             sha_state, | ||||||
|  |             entry_height, | ||||||
|  |             signature, | ||||||
|  |         }; | ||||||
|  |         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn new_advertise_recent_blockhash( | ||||||
|  |         from_keypair: &Keypair, | ||||||
|  |         storage_hash: Hash, | ||||||
|  |         recent_blockhash: Hash, | ||||||
|  |         entry_height: u64, | ||||||
|  |     ) -> Transaction { | ||||||
|  |         let program = StorageInstruction::AdvertiseStorageRecentBlockhash { | ||||||
|  |             hash: storage_hash, | ||||||
|  |             entry_height, | ||||||
|  |         }; | ||||||
|  |         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn new_proof_validation( | ||||||
|  |         from_keypair: &Keypair, | ||||||
|  |         recent_blockhash: Hash, | ||||||
|  |         entry_height: u64, | ||||||
|  |         proof_mask: Vec<ProofStatus>, | ||||||
|  |     ) -> Transaction { | ||||||
|  |         let program = StorageInstruction::ProofValidation { | ||||||
|  |             entry_height, | ||||||
|  |             proof_mask, | ||||||
|  |         }; | ||||||
|  |         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn new_reward_claim( | ||||||
|  |         from_keypair: &Keypair, | ||||||
|  |         recent_blockhash: Hash, | ||||||
|  |         entry_height: u64, | ||||||
|  |     ) -> Transaction { | ||||||
|  |         let program = StorageInstruction::ClaimStorageReward { entry_height }; | ||||||
|  |         Transaction::new_signed(from_keypair, &[], &id(), &program, recent_blockhash, 0) | ||||||
|  |     } | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user