icepick-solana: bugfix nondeterministic payer and simulate_transaction reporting

This commit is contained in:
Ryan Heywood 2025-02-18 02:26:13 -05:00
parent 3e9490644a
commit 3712fc2b51
Signed by: ryan
GPG Key ID: 8E401478A3FBEF72
4 changed files with 38 additions and 21 deletions

View File

@ -238,9 +238,11 @@ pub struct Inspect {
#[derive(Serialize, Deserialize, Debug)] #[derive(Serialize, Deserialize, Debug)]
pub struct Sign { pub struct Sign {
blockhash: String, blockhash: String,
transaction: solana_sdk::transaction::Transaction, instructions: Vec<solana_sdk::instruction::Instruction>,
#[serde(default)] #[serde(default)]
signing_keys: Vec<[u8; Keypair::SECRET_KEY_LENGTH]>, signing_keys: Vec<[u8; Keypair::SECRET_KEY_LENGTH]>,
#[serde(default)]
payer_address: Option<String>,
} }
#[derive(Serialize, Deserialize, Debug)] #[derive(Serialize, Deserialize, Debug)]
@ -987,9 +989,9 @@ impl Module for Solana {
derivation_accounts, derivation_accounts,
mut instructions, mut instructions,
}) => { }) => {
use solana_sdk::{hash::Hash, message::Message, transaction::Transaction}; use solana_sdk::hash::Hash;
let (hash, transaction) = match hashable { let hash = match hashable {
// We already have the account from GetNonceAccountData, // We already have the account from GetNonceAccountData,
// which also gives us the authority and the nonce itself. // which also gives us the authority and the nonce itself.
Hashable::Nonce { Hashable::Nonce {
@ -1005,21 +1007,14 @@ impl Module for Solana {
system_instruction::advance_nonce_account(&account_pk, &authority_pk); system_instruction::advance_nonce_account(&account_pk, &authority_pk);
instructions.insert(0, increment_nonce); instructions.insert(0, increment_nonce);
let message = Message::new(&instructions, None); hash
let transaction = Transaction::new_unsigned(message);
(hash, transaction)
}
Hashable::Blockhash { blockhash } => {
let blockhash = Hash::from_str(&blockhash).unwrap();
let message = Message::new(&instructions, None);
let transaction = Transaction::new_unsigned(message);
(blockhash, transaction)
} }
Hashable::Blockhash { blockhash } => Hash::from_str(&blockhash).unwrap(),
}; };
Ok(serde_json::json!({ Ok(serde_json::json!({
"blob": { "blob": {
"hash": hash, "hash": hash,
"transaction": transaction, "instructions": instructions,
}, },
"derivation_accounts": derivation_accounts, "derivation_accounts": derivation_accounts,
})) }))
@ -1035,9 +1030,12 @@ impl Module for Solana {
} }
Operation::Sign(Sign { Operation::Sign(Sign {
blockhash, blockhash,
mut transaction, instructions,
signing_keys, signing_keys,
payer_address,
}) => { }) => {
use solana_sdk::{message::Message, transaction::Transaction};
let keys = request let keys = request
.derived_keys .derived_keys
.unwrap_or_default() .unwrap_or_default()
@ -1046,10 +1044,21 @@ impl Module for Solana {
.map(|k| Self::keypair_from_bytes(*k)) .map(|k| Self::keypair_from_bytes(*k))
.collect::<Vec<_>>(); .collect::<Vec<_>>();
let payer_pk = payer_address
.as_deref()
.map(Pubkey::from_str)
.transpose()
.unwrap();
let message =
Message::new(&instructions, Some(&payer_pk.unwrap_or(keys[0].pubkey())));
let mut transaction = Transaction::new_unsigned(message);
let hash = solana_sdk::hash::Hash::from_str(&blockhash).unwrap(); let hash = solana_sdk::hash::Hash::from_str(&blockhash).unwrap();
transaction transaction
.try_sign(&keys, hash) .try_sign(&keys, hash)
.expect("not enough keys provided"); .expect("not enough keys provided");
Ok(serde_json::json!({ Ok(serde_json::json!({
"blob": { "blob": {
"transaction": transaction, "transaction": transaction,
@ -1065,7 +1074,15 @@ impl Module for Solana {
transaction.verify().expect("invalid signatures"); transaction.verify().expect("invalid signatures");
let client = solana_rpc_client::rpc_client::RpcClient::new(cluster_url); let client = solana_rpc_client::rpc_client::RpcClient::new(cluster_url);
let _simulated_response = client.simulate_transaction(&transaction).unwrap(); let simulated_response = client.simulate_transaction(&transaction).unwrap();
if let Some(err) = simulated_response.value.err {
return Ok(serde_json::json!({
"blob": {
"status": "simulate_transaction",
"error": err.to_string(),
}
}))
}
let response = client.send_and_confirm_transaction(&transaction); let response = client.send_and_confirm_transaction(&transaction);
let cluster_suffix = { let cluster_suffix = {
if cluster == Cluster::MainnetBeta { if cluster == Cluster::MainnetBeta {

View File

@ -46,12 +46,12 @@ step:
derivation_accounts: "derivation_accounts" derivation_accounts: "derivation_accounts"
blockhash: "blockhash" blockhash: "blockhash"
outputs: outputs:
transaction: "unsigned_transaction" instructions: "nonced_instructions"
- type: "sol-sign" - type: "sol-sign"
inputs: inputs:
blockhash: "blockhash" blockhash: "blockhash"
signing_keys: "private_keys" signing_keys: "private_keys"
transaction: "unsigned_transaction" instructions: "nonced_instructions"
outputs: outputs:
transaction: "signed_transaction" transaction: "signed_transaction"
- type: "sol-broadcast" - type: "sol-broadcast"

View File

@ -46,10 +46,10 @@ step:
nonce_authority: nonce_authority nonce_authority: nonce_authority
nonce_data: nonce_data nonce_data: nonce_data
outputs: outputs:
transaction: unsigned_transaction instructions: nonced_instructions
- type: sol-sign - type: sol-sign
inputs: inputs:
transaction: unsigned_transaction instructions: nonced_instructions
blockhash: nonce_data blockhash: nonce_data
outputs: outputs:
transaction: transaction transaction: transaction

View File

@ -35,11 +35,11 @@ step:
nonce_authority: "nonce_authority" nonce_authority: "nonce_authority"
nonce_data: "nonce_data" nonce_data: "nonce_data"
outputs: outputs:
transaction: "unsigned_transaction" instructions: "nonced_instructions"
- type: "sol-sign" - type: "sol-sign"
inputs: inputs:
blockhash: "nonce_data" blockhash: "nonce_data"
transaction: "unsigned_transaction" instructions: "nonced_instructions"
outputs: outputs:
transaction: "signed_transaction" transaction: "signed_transaction"
- type: "internal-save-file" - type: "internal-save-file"