178 lines
4.5 KiB
Rust
178 lines
4.5 KiB
Rust
use crate::{
|
|
accounts::{
|
|
ConfigAccount,
|
|
Endpoint,
|
|
EndpointDerivationData,
|
|
},
|
|
messages::{
|
|
GovernancePayloadUpgrade,
|
|
PayloadGovernanceRegisterChain,
|
|
},
|
|
types::*,
|
|
TokenBridgeError::{
|
|
InvalidChain,
|
|
InvalidGovernanceKey,
|
|
},
|
|
};
|
|
use bridge::{
|
|
vaa::{
|
|
ClaimableVAA,
|
|
DeserializePayload,
|
|
PayloadMessage,
|
|
},
|
|
CHAIN_ID_SOLANA,
|
|
};
|
|
use solana_program::{
|
|
account_info::AccountInfo,
|
|
program::invoke_signed,
|
|
program_error::ProgramError,
|
|
pubkey::Pubkey,
|
|
sysvar::{
|
|
clock::Clock,
|
|
rent::Rent,
|
|
},
|
|
};
|
|
use solitaire::{
|
|
processors::seeded::Seeded,
|
|
CreationLamports::Exempt,
|
|
*,
|
|
};
|
|
use std::ops::{
|
|
Deref,
|
|
DerefMut,
|
|
};
|
|
|
|
// Confirm that a ClaimableVAA came from the correct chain, signed by the right emitter.
|
|
fn verify_governance<'a, T>(vaa: &ClaimableVAA<'a, T>) -> Result<()>
|
|
where
|
|
T: DeserializePayload,
|
|
{
|
|
let expected_emitter = std::env!("EMITTER_ADDRESS");
|
|
let current_emitter = format!(
|
|
"{}",
|
|
Pubkey::new_from_array(vaa.message.meta().emitter_address)
|
|
);
|
|
// Fail if the emitter is not the known governance key, or the emitting chain is not Solana.
|
|
if expected_emitter != current_emitter || vaa.message.meta().emitter_chain != CHAIN_ID_SOLANA {
|
|
Err(InvalidGovernanceKey.into())
|
|
} else {
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(FromAccounts)]
|
|
pub struct UpgradeContract<'b> {
|
|
/// Payer for account creation (vaa-claim)
|
|
pub payer: Mut<Signer<Info<'b>>>,
|
|
|
|
/// GuardianSet change VAA
|
|
pub vaa: ClaimableVAA<'b, GovernancePayloadUpgrade>,
|
|
|
|
/// PDA authority for the loader
|
|
pub upgrade_authority: Derive<Info<'b>, "upgrade">,
|
|
|
|
/// Spill address for the upgrade excess lamports
|
|
pub spill: Mut<Info<'b>>,
|
|
|
|
/// New contract address.
|
|
pub buffer: Mut<Info<'b>>,
|
|
|
|
/// Required by the upgradeable uploader.
|
|
pub program_data: Mut<Info<'b>>,
|
|
|
|
/// Our own address, required by the upgradeable loader.
|
|
pub own_address: Mut<Info<'b>>,
|
|
|
|
// Various sysvar/program accounts needed for the upgradeable loader.
|
|
pub rent: Sysvar<'b, Rent>,
|
|
pub clock: Sysvar<'b, Clock>,
|
|
pub bpf_loader: Info<'b>,
|
|
pub system: Info<'b>,
|
|
}
|
|
|
|
impl<'b> InstructionContext<'b> for UpgradeContract<'b> {
|
|
}
|
|
|
|
#[derive(BorshDeserialize, BorshSerialize, Default)]
|
|
pub struct UpgradeContractData {}
|
|
|
|
pub fn upgrade_contract(
|
|
ctx: &ExecutionContext,
|
|
accs: &mut UpgradeContract,
|
|
_data: UpgradeContractData,
|
|
) -> Result<()> {
|
|
verify_governance(&accs.vaa)?;
|
|
accs.vaa.verify(&ctx.program_id)?;
|
|
|
|
accs.vaa.claim(ctx, accs.payer.key)?;
|
|
|
|
let upgrade_ix = solana_program::bpf_loader_upgradeable::upgrade(
|
|
ctx.program_id,
|
|
&accs.vaa.message.new_contract,
|
|
accs.upgrade_authority.key,
|
|
accs.spill.key,
|
|
);
|
|
|
|
let seeds = accs
|
|
.upgrade_authority
|
|
.self_bumped_seeds(None, ctx.program_id);
|
|
let seeds: Vec<&[u8]> = seeds.iter().map(|item| item.as_slice()).collect();
|
|
let seeds = seeds.as_slice();
|
|
invoke_signed(&upgrade_ix, ctx.accounts, &[seeds])?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[derive(FromAccounts)]
|
|
pub struct RegisterChain<'b> {
|
|
pub payer: Mut<Signer<AccountInfo<'b>>>,
|
|
pub config: ConfigAccount<'b, { AccountState::Initialized }>,
|
|
|
|
pub endpoint: Mut<Endpoint<'b, { AccountState::Uninitialized }>>,
|
|
|
|
pub vaa: ClaimableVAA<'b, PayloadGovernanceRegisterChain>,
|
|
}
|
|
|
|
impl<'a> From<&RegisterChain<'a>> for EndpointDerivationData {
|
|
fn from(accs: &RegisterChain<'a>) -> Self {
|
|
EndpointDerivationData {
|
|
emitter_chain: accs.vaa.chain,
|
|
emitter_address: accs.vaa.endpoint_address,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'b> InstructionContext<'b> for RegisterChain<'b> {
|
|
}
|
|
|
|
#[derive(BorshDeserialize, BorshSerialize, Default)]
|
|
pub struct RegisterChainData {}
|
|
|
|
pub fn register_chain(
|
|
ctx: &ExecutionContext,
|
|
accs: &mut RegisterChain,
|
|
data: RegisterChainData,
|
|
) -> Result<()> {
|
|
let derivation_data: EndpointDerivationData = (&*accs).into();
|
|
accs.endpoint
|
|
.verify_derivation(ctx.program_id, &derivation_data)?;
|
|
|
|
// Claim VAA
|
|
verify_governance(&accs.vaa)?;
|
|
accs.vaa.verify(&ctx.program_id)?;
|
|
accs.vaa.claim(ctx, accs.payer.key)?;
|
|
|
|
if accs.vaa.chain == CHAIN_ID_SOLANA {
|
|
return Err(InvalidChain.into());
|
|
}
|
|
|
|
// Create endpoint
|
|
accs.endpoint
|
|
.create(&((&*accs).into()), ctx, accs.payer.key, Exempt);
|
|
|
|
accs.endpoint.chain = accs.vaa.chain;
|
|
accs.endpoint.contract = accs.vaa.endpoint_address;
|
|
|
|
Ok(())
|
|
}
|