2019-11-06 14:15:00 -07:00
|
|
|
use serde_json::Value;
|
2020-12-21 13:02:53 -08:00
|
|
|
use solana_cli::{
|
|
|
|
cli::{process_command, CliCommand, CliConfig},
|
|
|
|
program::ProgramCliCommand,
|
|
|
|
};
|
2019-03-16 22:37:20 -07:00
|
|
|
use solana_client::rpc_client::RpcClient;
|
2020-09-18 22:21:44 -07:00
|
|
|
use solana_core::test_validator::TestValidator;
|
2019-12-16 14:05:17 -07:00
|
|
|
use solana_faucet::faucet::run_local_faucet;
|
2020-06-04 20:14:12 -06:00
|
|
|
use solana_sdk::{
|
2021-01-08 09:37:57 -08:00
|
|
|
account_utils::StateMut,
|
2020-06-04 20:14:12 -06:00
|
|
|
bpf_loader,
|
2020-12-14 15:35:10 -08:00
|
|
|
bpf_loader_upgradeable::{self, UpgradeableLoaderState},
|
2020-06-17 12:18:48 -06:00
|
|
|
commitment_config::CommitmentConfig,
|
2020-06-04 20:14:12 -06:00
|
|
|
pubkey::Pubkey,
|
|
|
|
signature::{Keypair, Signer},
|
|
|
|
};
|
2020-11-25 17:00:47 -08:00
|
|
|
use std::{fs::File, io::Read, path::PathBuf, str::FromStr, sync::mpsc::channel};
|
2019-01-17 09:49:32 -07:00
|
|
|
|
|
|
|
#[test]
|
2020-12-21 13:02:53 -08:00
|
|
|
fn test_cli_program_deploy_non_upgradeable() {
|
2019-02-05 08:03:52 -08:00
|
|
|
solana_logger::setup();
|
|
|
|
|
2019-01-17 09:49:32 -07:00
|
|
|
let mut pathbuf = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
pathbuf.push("tests");
|
|
|
|
pathbuf.push("fixtures");
|
|
|
|
pathbuf.push("noop");
|
|
|
|
pathbuf.set_extension("so");
|
|
|
|
|
2020-12-08 23:18:27 -08:00
|
|
|
let mint_keypair = Keypair::new();
|
|
|
|
let test_validator = TestValidator::with_no_fees(mint_keypair.pubkey());
|
2019-01-17 09:49:32 -07:00
|
|
|
|
|
|
|
let (sender, receiver) = channel();
|
2020-12-08 23:18:27 -08:00
|
|
|
run_local_faucet(mint_keypair, sender, None);
|
2019-12-16 14:05:17 -07:00
|
|
|
let faucet_addr = receiver.recv().unwrap();
|
2019-01-17 09:49:32 -07:00
|
|
|
|
2020-11-25 17:00:47 -08:00
|
|
|
let rpc_client = RpcClient::new(test_validator.rpc_url());
|
2019-01-17 09:49:32 -07:00
|
|
|
|
2019-10-01 01:14:49 +05:30
|
|
|
let mut file = File::open(pathbuf.to_str().unwrap()).unwrap();
|
|
|
|
let mut program_data = Vec::new();
|
|
|
|
file.read_to_end(&mut program_data).unwrap();
|
|
|
|
let minimum_balance_for_rent_exemption = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(program_data.len())
|
|
|
|
.unwrap();
|
|
|
|
|
2020-06-17 12:18:48 -06:00
|
|
|
let mut config = CliConfig::recent_for_tests();
|
2020-02-24 17:03:30 -07:00
|
|
|
let keypair = Keypair::new();
|
2020-11-25 17:00:47 -08:00
|
|
|
config.json_rpc_url = test_validator.rpc_url();
|
2020-12-21 13:02:53 -08:00
|
|
|
config.signers = vec![&keypair];
|
2019-10-04 16:13:21 -06:00
|
|
|
config.command = CliCommand::Airdrop {
|
2019-12-16 14:05:17 -07:00
|
|
|
faucet_host: None,
|
|
|
|
faucet_port: faucet_addr.port(),
|
2020-02-16 11:41:00 -07:00
|
|
|
pubkey: None,
|
2020-10-22 22:42:35 -06:00
|
|
|
lamports: 4 * minimum_balance_for_rent_exemption, // min balance for rent exemption for three programs + leftover for tx processing
|
2019-08-29 20:45:53 -07:00
|
|
|
};
|
2019-01-28 14:52:35 -08:00
|
|
|
process_command(&config).unwrap();
|
2019-01-17 09:49:32 -07:00
|
|
|
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Deploy {
|
2020-06-04 20:14:12 -06:00
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
2020-12-21 13:02:53 -08:00
|
|
|
address: None,
|
2020-08-17 18:06:22 -07:00
|
|
|
use_deprecated_loader: false,
|
2020-10-22 22:42:35 -06:00
|
|
|
allow_excessive_balance: false,
|
2020-06-04 20:14:12 -06:00
|
|
|
};
|
2019-01-17 09:49:32 -07:00
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let program_id_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
2021-01-08 09:37:57 -08:00
|
|
|
.get("ProgramId")
|
2019-01-17 09:49:32 -07:00
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
2019-11-06 14:15:00 -07:00
|
|
|
let program_id = Pubkey::from_str(&program_id_str).unwrap();
|
2020-06-17 12:18:48 -06:00
|
|
|
let account0 = rpc_client
|
|
|
|
.get_account_with_commitment(&program_id, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
2020-06-04 20:14:12 -06:00
|
|
|
assert_eq!(account0.lamports, minimum_balance_for_rent_exemption);
|
|
|
|
assert_eq!(account0.owner, bpf_loader::id());
|
|
|
|
assert_eq!(account0.executable, true);
|
2019-01-17 09:49:32 -07:00
|
|
|
let mut file = File::open(pathbuf.to_str().unwrap().to_string()).unwrap();
|
|
|
|
let mut elf = Vec::new();
|
|
|
|
file.read_to_end(&mut elf).unwrap();
|
2020-06-04 20:14:12 -06:00
|
|
|
assert_eq!(account0.data, elf);
|
|
|
|
|
|
|
|
// Test custom address
|
|
|
|
let custom_address_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &custom_address_keypair];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Deploy {
|
2020-06-04 20:14:12 -06:00
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
2020-12-21 13:02:53 -08:00
|
|
|
address: Some(1),
|
2020-08-17 18:06:22 -07:00
|
|
|
use_deprecated_loader: false,
|
2020-10-22 22:42:35 -06:00
|
|
|
allow_excessive_balance: false,
|
2020-06-04 20:14:12 -06:00
|
|
|
};
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
let account1 = rpc_client
|
2020-06-17 12:18:48 -06:00
|
|
|
.get_account_with_commitment(&custom_address_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
2020-06-04 20:14:12 -06:00
|
|
|
.unwrap();
|
|
|
|
assert_eq!(account1.lamports, minimum_balance_for_rent_exemption);
|
|
|
|
assert_eq!(account1.owner, bpf_loader::id());
|
|
|
|
assert_eq!(account1.executable, true);
|
2020-12-21 13:02:53 -08:00
|
|
|
assert_eq!(account1.data, account0.data);
|
2020-06-04 20:14:12 -06:00
|
|
|
|
|
|
|
// Attempt to redeploy to the same address
|
|
|
|
process_command(&config).unwrap_err();
|
2019-01-17 09:49:32 -07:00
|
|
|
|
2020-10-22 22:42:35 -06:00
|
|
|
// Attempt to deploy to account with excess balance
|
|
|
|
let custom_address_keypair = Keypair::new();
|
2020-12-21 13:02:53 -08:00
|
|
|
config.signers = vec![&custom_address_keypair];
|
2020-10-22 22:42:35 -06:00
|
|
|
config.command = CliCommand::Airdrop {
|
|
|
|
faucet_host: None,
|
|
|
|
faucet_port: faucet_addr.port(),
|
|
|
|
pubkey: None,
|
|
|
|
lamports: 2 * minimum_balance_for_rent_exemption, // Anything over minimum_balance_for_rent_exemption should trigger err
|
|
|
|
};
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
config.signers = vec![&keypair, &custom_address_keypair];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Deploy {
|
2020-10-22 22:42:35 -06:00
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
2020-12-21 13:02:53 -08:00
|
|
|
address: Some(1),
|
2020-10-22 22:42:35 -06:00
|
|
|
use_deprecated_loader: false,
|
|
|
|
allow_excessive_balance: false,
|
|
|
|
};
|
|
|
|
process_command(&config).unwrap_err();
|
|
|
|
|
|
|
|
// Use forcing parameter to deploy to account with excess balance
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Deploy {
|
2020-10-22 22:42:35 -06:00
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
2020-12-21 13:02:53 -08:00
|
|
|
address: Some(1),
|
2020-10-22 22:42:35 -06:00
|
|
|
use_deprecated_loader: false,
|
|
|
|
allow_excessive_balance: true,
|
|
|
|
};
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
let account2 = rpc_client
|
|
|
|
.get_account_with_commitment(&custom_address_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(account2.lamports, 2 * minimum_balance_for_rent_exemption);
|
|
|
|
assert_eq!(account2.owner, bpf_loader::id());
|
|
|
|
assert_eq!(account2.executable, true);
|
2020-12-21 13:02:53 -08:00
|
|
|
assert_eq!(account2.data, account0.data);
|
2019-01-17 09:49:32 -07:00
|
|
|
}
|
2020-12-14 15:35:10 -08:00
|
|
|
|
|
|
|
#[test]
|
2020-12-21 13:02:53 -08:00
|
|
|
fn test_cli_program_deploy_no_authority() {
|
2020-12-14 15:35:10 -08:00
|
|
|
solana_logger::setup();
|
|
|
|
|
|
|
|
let mut pathbuf = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
pathbuf.push("tests");
|
|
|
|
pathbuf.push("fixtures");
|
|
|
|
pathbuf.push("noop");
|
|
|
|
pathbuf.set_extension("so");
|
|
|
|
|
|
|
|
let mint_keypair = Keypair::new();
|
|
|
|
let test_validator = TestValidator::with_no_fees(mint_keypair.pubkey());
|
|
|
|
|
|
|
|
let (sender, receiver) = channel();
|
|
|
|
run_local_faucet(mint_keypair, sender, None);
|
|
|
|
let faucet_addr = receiver.recv().unwrap();
|
|
|
|
|
|
|
|
let rpc_client = RpcClient::new(test_validator.rpc_url());
|
|
|
|
|
|
|
|
let mut file = File::open(pathbuf.to_str().unwrap()).unwrap();
|
|
|
|
let mut program_data = Vec::new();
|
|
|
|
file.read_to_end(&mut program_data).unwrap();
|
|
|
|
let max_len = program_data.len();
|
|
|
|
let minimum_balance_for_programdata = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(
|
|
|
|
UpgradeableLoaderState::programdata_len(max_len).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
let minimum_balance_for_program = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(UpgradeableLoaderState::program_len().unwrap())
|
|
|
|
.unwrap();
|
|
|
|
let upgrade_authority = Keypair::new();
|
|
|
|
|
|
|
|
let mut config = CliConfig::recent_for_tests();
|
|
|
|
let keypair = Keypair::new();
|
|
|
|
config.json_rpc_url = test_validator.rpc_url();
|
|
|
|
config.command = CliCommand::Airdrop {
|
|
|
|
faucet_host: None,
|
|
|
|
faucet_port: faucet_addr.port(),
|
|
|
|
pubkey: None,
|
|
|
|
lamports: 100 * minimum_balance_for_programdata + minimum_balance_for_program,
|
|
|
|
};
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
|
2020-12-21 13:02:53 -08:00
|
|
|
// Deploy a program with no authority
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: None,
|
|
|
|
program_pubkey: None,
|
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-14 15:35:10 -08:00
|
|
|
allow_excessive_balance: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
upgrade_authority_signer_index: None,
|
|
|
|
upgrade_authority_pubkey: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
max_len: None,
|
|
|
|
});
|
2020-12-14 15:35:10 -08:00
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let program_id_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
2021-01-08 09:37:57 -08:00
|
|
|
.get("ProgramId")
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
let program_id = Pubkey::from_str(&program_id_str).unwrap();
|
|
|
|
|
2020-12-21 13:02:53 -08:00
|
|
|
// Attempt to upgrade the program
|
2020-12-14 15:35:10 -08:00
|
|
|
config.signers = vec![&keypair, &upgrade_authority];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: None,
|
|
|
|
program_pubkey: Some(program_id),
|
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-21 13:02:53 -08:00
|
|
|
allow_excessive_balance: false,
|
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(upgrade_authority.pubkey()),
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
max_len: None,
|
|
|
|
});
|
2020-12-14 15:35:10 -08:00
|
|
|
process_command(&config).unwrap_err();
|
2020-12-21 13:02:53 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_cli_program_deploy_with_authority() {
|
|
|
|
solana_logger::setup();
|
|
|
|
|
|
|
|
let mut pathbuf = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
pathbuf.push("tests");
|
|
|
|
pathbuf.push("fixtures");
|
|
|
|
pathbuf.push("noop");
|
|
|
|
pathbuf.set_extension("so");
|
|
|
|
|
|
|
|
let mint_keypair = Keypair::new();
|
|
|
|
let test_validator = TestValidator::with_no_fees(mint_keypair.pubkey());
|
|
|
|
|
|
|
|
let (sender, receiver) = channel();
|
|
|
|
run_local_faucet(mint_keypair, sender, None);
|
|
|
|
let faucet_addr = receiver.recv().unwrap();
|
|
|
|
|
|
|
|
let rpc_client = RpcClient::new(test_validator.rpc_url());
|
|
|
|
|
|
|
|
let mut file = File::open(pathbuf.to_str().unwrap()).unwrap();
|
|
|
|
let mut program_data = Vec::new();
|
|
|
|
file.read_to_end(&mut program_data).unwrap();
|
|
|
|
let max_len = program_data.len();
|
|
|
|
let minimum_balance_for_programdata = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(
|
|
|
|
UpgradeableLoaderState::programdata_len(max_len).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
let minimum_balance_for_program = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(UpgradeableLoaderState::program_len().unwrap())
|
|
|
|
.unwrap();
|
|
|
|
let upgrade_authority = Keypair::new();
|
|
|
|
|
|
|
|
let mut config = CliConfig::recent_for_tests();
|
|
|
|
let keypair = Keypair::new();
|
|
|
|
config.json_rpc_url = test_validator.rpc_url();
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Airdrop {
|
|
|
|
faucet_host: None,
|
|
|
|
faucet_port: faucet_addr.port(),
|
|
|
|
pubkey: None,
|
|
|
|
lamports: 100 * minimum_balance_for_programdata + minimum_balance_for_program,
|
|
|
|
};
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
|
|
|
|
// Deploy the upgradeable program with specified program_id
|
|
|
|
let program_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &upgrade_authority, &program_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: Some(2),
|
|
|
|
program_pubkey: Some(program_keypair.pubkey()),
|
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-21 13:02:53 -08:00
|
|
|
allow_excessive_balance: false,
|
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(upgrade_authority.pubkey()),
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
max_len: Some(max_len),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
2021-01-08 09:37:57 -08:00
|
|
|
let program_pubkey_str = json
|
2020-12-21 13:02:53 -08:00
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
2021-01-08 09:37:57 -08:00
|
|
|
.get("ProgramId")
|
2020-12-21 13:02:53 -08:00
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
program_keypair.pubkey(),
|
2021-01-08 09:37:57 -08:00
|
|
|
Pubkey::from_str(&program_pubkey_str).unwrap()
|
2020-12-21 13:02:53 -08:00
|
|
|
);
|
|
|
|
let program_account = rpc_client
|
|
|
|
.get_account_with_commitment(&program_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(program_account.lamports, minimum_balance_for_program);
|
|
|
|
assert_eq!(program_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(program_account.executable, true);
|
|
|
|
let (programdata_pubkey, _) = Pubkey::find_program_address(
|
|
|
|
&[program_keypair.pubkey().as_ref()],
|
|
|
|
&bpf_loader_upgradeable::id(),
|
|
|
|
);
|
|
|
|
let programdata_account = rpc_client
|
|
|
|
.get_account_with_commitment(&programdata_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.lamports,
|
|
|
|
minimum_balance_for_programdata
|
|
|
|
);
|
|
|
|
assert_eq!(programdata_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(programdata_account.executable, false);
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.data[UpgradeableLoaderState::programdata_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
2020-12-14 15:35:10 -08:00
|
|
|
|
|
|
|
// Deploy the upgradeable program
|
2020-12-21 13:02:53 -08:00
|
|
|
config.signers = vec![&keypair, &upgrade_authority];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: None,
|
|
|
|
program_pubkey: None,
|
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-14 15:35:10 -08:00
|
|
|
allow_excessive_balance: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(upgrade_authority.pubkey()),
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-14 15:35:10 -08:00
|
|
|
max_len: Some(max_len),
|
2020-12-21 13:02:53 -08:00
|
|
|
});
|
2020-12-14 15:35:10 -08:00
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
2021-01-08 09:37:57 -08:00
|
|
|
let program_pubkey_str = json
|
2020-12-14 15:35:10 -08:00
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
2021-01-08 09:37:57 -08:00
|
|
|
.get("ProgramId")
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
2021-01-08 09:37:57 -08:00
|
|
|
let program_pubkey = Pubkey::from_str(&program_pubkey_str).unwrap();
|
2020-12-14 15:35:10 -08:00
|
|
|
let program_account = rpc_client
|
2021-01-08 09:37:57 -08:00
|
|
|
.get_account_with_commitment(&program_pubkey, CommitmentConfig::recent())
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(program_account.lamports, minimum_balance_for_program);
|
|
|
|
assert_eq!(program_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(program_account.executable, true);
|
|
|
|
let (programdata_pubkey, _) =
|
2021-01-08 09:37:57 -08:00
|
|
|
Pubkey::find_program_address(&[program_pubkey.as_ref()], &bpf_loader_upgradeable::id());
|
2020-12-14 15:35:10 -08:00
|
|
|
let programdata_account = rpc_client
|
|
|
|
.get_account_with_commitment(&programdata_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.lamports,
|
|
|
|
minimum_balance_for_programdata
|
|
|
|
);
|
|
|
|
assert_eq!(programdata_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(programdata_account.executable, false);
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.data[UpgradeableLoaderState::programdata_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Upgrade the program
|
|
|
|
config.signers = vec![&keypair, &upgrade_authority];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
program_pubkey: Some(program_pubkey),
|
2020-12-21 13:02:53 -08:00
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-21 13:02:53 -08:00
|
|
|
allow_excessive_balance: false,
|
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(upgrade_authority.pubkey()),
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
max_len: Some(max_len),
|
|
|
|
});
|
2021-01-08 09:37:57 -08:00
|
|
|
process_command(&config).unwrap();
|
2020-12-14 15:35:10 -08:00
|
|
|
let program_account = rpc_client
|
2021-01-08 09:37:57 -08:00
|
|
|
.get_account_with_commitment(&program_pubkey, CommitmentConfig::recent())
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(program_account.lamports, minimum_balance_for_program);
|
|
|
|
assert_eq!(program_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(program_account.executable, true);
|
|
|
|
let (programdata_pubkey, _) =
|
2021-01-08 09:37:57 -08:00
|
|
|
Pubkey::find_program_address(&[program_pubkey.as_ref()], &bpf_loader_upgradeable::id());
|
2020-12-14 15:35:10 -08:00
|
|
|
let programdata_account = rpc_client
|
|
|
|
.get_account_with_commitment(&programdata_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.lamports,
|
|
|
|
minimum_balance_for_programdata
|
|
|
|
);
|
|
|
|
assert_eq!(programdata_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(programdata_account.executable, false);
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.data[UpgradeableLoaderState::programdata_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Set a new authority
|
|
|
|
let new_upgrade_authority = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &upgrade_authority];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::SetUpgradeAuthority {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_pubkey,
|
2020-12-21 13:02:53 -08:00
|
|
|
upgrade_authority_index: Some(1),
|
2020-12-14 15:35:10 -08:00
|
|
|
new_upgrade_authority: Some(new_upgrade_authority.pubkey()),
|
2020-12-21 13:02:53 -08:00
|
|
|
});
|
2020-12-14 15:35:10 -08:00
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let new_upgrade_authority_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
2021-01-08 09:37:57 -08:00
|
|
|
.get("Authority")
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
Pubkey::from_str(&new_upgrade_authority_str).unwrap(),
|
|
|
|
new_upgrade_authority.pubkey()
|
|
|
|
);
|
|
|
|
|
|
|
|
// Upgrade with new authority
|
|
|
|
config.signers = vec![&keypair, &new_upgrade_authority];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
program_pubkey: Some(program_pubkey),
|
2020-12-21 13:02:53 -08:00
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-21 13:02:53 -08:00
|
|
|
allow_excessive_balance: false,
|
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(new_upgrade_authority.pubkey()),
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
max_len: None,
|
|
|
|
});
|
2021-01-08 09:37:57 -08:00
|
|
|
process_command(&config).unwrap();
|
2020-12-14 15:35:10 -08:00
|
|
|
let program_account = rpc_client
|
2021-01-08 09:37:57 -08:00
|
|
|
.get_account_with_commitment(&program_pubkey, CommitmentConfig::recent())
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(program_account.lamports, minimum_balance_for_program);
|
|
|
|
assert_eq!(program_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(program_account.executable, true);
|
|
|
|
let (programdata_pubkey, _) =
|
2021-01-08 09:37:57 -08:00
|
|
|
Pubkey::find_program_address(&[program_pubkey.as_ref()], &bpf_loader_upgradeable::id());
|
2020-12-14 15:35:10 -08:00
|
|
|
let programdata_account = rpc_client
|
|
|
|
.get_account_with_commitment(&programdata_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.lamports,
|
|
|
|
minimum_balance_for_programdata
|
|
|
|
);
|
|
|
|
assert_eq!(programdata_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
assert_eq!(programdata_account.executable, false);
|
|
|
|
assert_eq!(
|
|
|
|
programdata_account.data[UpgradeableLoaderState::programdata_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
2021-01-08 14:43:36 -08:00
|
|
|
// Get upgrade authority
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::GetAuthority {
|
|
|
|
account_pubkey: Some(program_pubkey),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let authority_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Program upgrade authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
new_upgrade_authority.pubkey(),
|
|
|
|
Pubkey::from_str(&authority_pubkey_str).unwrap()
|
|
|
|
);
|
|
|
|
|
2020-12-21 13:02:53 -08:00
|
|
|
// Set no authority
|
2020-12-14 15:35:10 -08:00
|
|
|
config.signers = vec![&keypair, &new_upgrade_authority];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::SetUpgradeAuthority {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_pubkey,
|
2020-12-21 13:02:53 -08:00
|
|
|
upgrade_authority_index: Some(1),
|
2020-12-14 15:35:10 -08:00
|
|
|
new_upgrade_authority: None,
|
2020-12-21 13:02:53 -08:00
|
|
|
});
|
2020-12-14 15:35:10 -08:00
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let new_upgrade_authority_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
2021-01-08 09:37:57 -08:00
|
|
|
.get("Authority")
|
2020-12-14 15:35:10 -08:00
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(new_upgrade_authority_str, "None");
|
|
|
|
|
|
|
|
// Upgrade with no authority
|
|
|
|
config.signers = vec![&keypair, &new_upgrade_authority];
|
2020-12-21 13:02:53 -08:00
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
2021-01-08 09:37:57 -08:00
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
2020-12-21 13:02:53 -08:00
|
|
|
program_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
program_pubkey: Some(program_pubkey),
|
2020-12-21 13:02:53 -08:00
|
|
|
buffer_signer_index: None,
|
2021-01-08 09:37:57 -08:00
|
|
|
buffer_pubkey: None,
|
2020-12-21 13:02:53 -08:00
|
|
|
allow_excessive_balance: false,
|
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(new_upgrade_authority.pubkey()),
|
2021-01-08 09:37:57 -08:00
|
|
|
is_final: false,
|
2020-12-21 13:02:53 -08:00
|
|
|
max_len: None,
|
|
|
|
});
|
2020-12-14 15:35:10 -08:00
|
|
|
process_command(&config).unwrap_err();
|
2021-01-08 09:37:57 -08:00
|
|
|
|
|
|
|
// deploy with finality
|
|
|
|
config.signers = vec![&keypair, &new_upgrade_authority];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::Deploy {
|
|
|
|
program_location: Some(pathbuf.to_str().unwrap().to_string()),
|
|
|
|
program_signer_index: None,
|
|
|
|
program_pubkey: None,
|
|
|
|
buffer_signer_index: None,
|
|
|
|
buffer_pubkey: None,
|
|
|
|
allow_excessive_balance: false,
|
|
|
|
upgrade_authority_signer_index: Some(1),
|
|
|
|
upgrade_authority_pubkey: Some(new_upgrade_authority.pubkey()),
|
|
|
|
is_final: true,
|
|
|
|
max_len: None,
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let program_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("ProgramId")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
let program_pubkey = Pubkey::from_str(&program_pubkey_str).unwrap();
|
|
|
|
let (programdata_pubkey, _) =
|
|
|
|
Pubkey::find_program_address(&[program_pubkey.as_ref()], &bpf_loader_upgradeable::id());
|
|
|
|
let programdata_account = rpc_client
|
|
|
|
.get_account_with_commitment(&programdata_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
if let UpgradeableLoaderState::ProgramData {
|
|
|
|
slot: _,
|
|
|
|
upgrade_authority_address,
|
|
|
|
} = programdata_account.state().unwrap()
|
|
|
|
{
|
|
|
|
assert_eq!(upgrade_authority_address, None);
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
2021-01-08 14:43:36 -08:00
|
|
|
|
|
|
|
// Get buffer authority
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::GetAuthority {
|
|
|
|
account_pubkey: Some(program_pubkey),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let authority_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Program upgrade authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!("None", authority_pubkey_str);
|
2021-01-08 09:37:57 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_cli_program_write_buffer() {
|
|
|
|
solana_logger::setup();
|
|
|
|
|
|
|
|
let mut pathbuf = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
pathbuf.push("tests");
|
|
|
|
pathbuf.push("fixtures");
|
|
|
|
pathbuf.push("noop");
|
|
|
|
pathbuf.set_extension("so");
|
|
|
|
|
|
|
|
let mint_keypair = Keypair::new();
|
|
|
|
let test_validator = TestValidator::with_no_fees(mint_keypair.pubkey());
|
|
|
|
|
|
|
|
let (sender, receiver) = channel();
|
|
|
|
run_local_faucet(mint_keypair, sender, None);
|
|
|
|
let faucet_addr = receiver.recv().unwrap();
|
|
|
|
|
|
|
|
let rpc_client = RpcClient::new(test_validator.rpc_url());
|
|
|
|
|
|
|
|
let mut file = File::open(pathbuf.to_str().unwrap()).unwrap();
|
|
|
|
let mut program_data = Vec::new();
|
|
|
|
file.read_to_end(&mut program_data).unwrap();
|
|
|
|
let max_len = program_data.len();
|
|
|
|
let minimum_balance_for_buffer = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(
|
|
|
|
UpgradeableLoaderState::programdata_len(max_len).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
let minimum_balance_for_buffer_default = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(
|
|
|
|
UpgradeableLoaderState::programdata_len(max_len * 2).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let mut config = CliConfig::recent_for_tests();
|
|
|
|
let keypair = Keypair::new();
|
|
|
|
config.json_rpc_url = test_validator.rpc_url();
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Airdrop {
|
|
|
|
faucet_host: None,
|
|
|
|
faucet_port: faucet_addr.port(),
|
|
|
|
pubkey: None,
|
|
|
|
lamports: 100 * minimum_balance_for_buffer,
|
|
|
|
};
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
|
|
|
|
// Write a buffer with default params
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::WriteBuffer {
|
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
|
|
|
buffer_signer_index: None,
|
|
|
|
buffer_pubkey: None,
|
|
|
|
buffer_authority_signer_index: None,
|
|
|
|
is_final: false,
|
|
|
|
max_len: None,
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
let new_buffer_pubkey = Pubkey::from_str(&buffer_pubkey_str).unwrap();
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&new_buffer_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(buffer_account.lamports, minimum_balance_for_buffer_default);
|
|
|
|
assert_eq!(buffer_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(keypair.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
assert_eq!(
|
|
|
|
buffer_account.data[UpgradeableLoaderState::buffer_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Specify buffer keypair and max_len
|
|
|
|
let buffer_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::WriteBuffer {
|
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
|
|
|
buffer_signer_index: Some(1),
|
|
|
|
buffer_pubkey: Some(buffer_keypair.pubkey()),
|
|
|
|
buffer_authority_signer_index: None,
|
|
|
|
is_final: false,
|
|
|
|
max_len: Some(max_len),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
buffer_keypair.pubkey(),
|
|
|
|
Pubkey::from_str(&buffer_pubkey_str).unwrap()
|
|
|
|
);
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(buffer_account.lamports, minimum_balance_for_buffer);
|
|
|
|
assert_eq!(buffer_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(keypair.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
assert_eq!(
|
|
|
|
buffer_account.data[UpgradeableLoaderState::buffer_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
2021-01-08 14:43:36 -08:00
|
|
|
// Get buffer authority
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::GetAuthority {
|
|
|
|
account_pubkey: Some(buffer_keypair.pubkey()),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let authority_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
keypair.pubkey(),
|
|
|
|
Pubkey::from_str(&authority_pubkey_str).unwrap()
|
|
|
|
);
|
|
|
|
|
2021-01-08 09:37:57 -08:00
|
|
|
// Specify buffer authority
|
|
|
|
let buffer_keypair = Keypair::new();
|
|
|
|
let authority_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair, &authority_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::WriteBuffer {
|
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
|
|
|
buffer_signer_index: Some(1),
|
|
|
|
buffer_pubkey: Some(buffer_keypair.pubkey()),
|
|
|
|
buffer_authority_signer_index: Some(2),
|
|
|
|
is_final: false,
|
|
|
|
max_len: None,
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
buffer_keypair.pubkey(),
|
|
|
|
Pubkey::from_str(&buffer_pubkey_str).unwrap()
|
|
|
|
);
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(buffer_account.lamports, minimum_balance_for_buffer_default);
|
|
|
|
assert_eq!(buffer_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(authority_keypair.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
assert_eq!(
|
|
|
|
buffer_account.data[UpgradeableLoaderState::buffer_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Specify authority only
|
|
|
|
let buffer_keypair = Keypair::new();
|
|
|
|
let authority_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair, &authority_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::WriteBuffer {
|
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
|
|
|
buffer_signer_index: None,
|
|
|
|
buffer_pubkey: None,
|
|
|
|
buffer_authority_signer_index: Some(2),
|
|
|
|
is_final: false,
|
|
|
|
max_len: None,
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
let buffer_pubkey = Pubkey::from_str(&buffer_pubkey_str).unwrap();
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(buffer_account.lamports, minimum_balance_for_buffer_default);
|
|
|
|
assert_eq!(buffer_account.owner, bpf_loader_upgradeable::id());
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(authority_keypair.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
assert_eq!(
|
|
|
|
buffer_account.data[UpgradeableLoaderState::buffer_data_offset().unwrap()..],
|
|
|
|
program_data[..]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Specify final
|
|
|
|
let buffer_keypair = Keypair::new();
|
|
|
|
let authority_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair, &authority_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::WriteBuffer {
|
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
|
|
|
buffer_signer_index: None,
|
|
|
|
buffer_pubkey: None,
|
|
|
|
buffer_authority_signer_index: Some(2),
|
|
|
|
is_final: true,
|
|
|
|
max_len: None,
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
let buffer_pubkey = Pubkey::from_str(&buffer_pubkey_str).unwrap();
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_pubkey, CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, None);
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
2021-01-08 14:43:36 -08:00
|
|
|
|
|
|
|
// Get buffer authority
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::GetAuthority {
|
|
|
|
account_pubkey: Some(buffer_pubkey),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let authority_pubkey_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Buffer authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!("None", authority_pubkey_str);
|
2021-01-08 09:37:57 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_cli_program_set_buffer_authority() {
|
|
|
|
solana_logger::setup();
|
|
|
|
|
|
|
|
let mut pathbuf = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
pathbuf.push("tests");
|
|
|
|
pathbuf.push("fixtures");
|
|
|
|
pathbuf.push("noop");
|
|
|
|
pathbuf.set_extension("so");
|
|
|
|
|
|
|
|
let mint_keypair = Keypair::new();
|
|
|
|
let test_validator = TestValidator::with_no_fees(mint_keypair.pubkey());
|
|
|
|
|
|
|
|
let (sender, receiver) = channel();
|
|
|
|
run_local_faucet(mint_keypair, sender, None);
|
|
|
|
let faucet_addr = receiver.recv().unwrap();
|
|
|
|
|
|
|
|
let rpc_client = RpcClient::new(test_validator.rpc_url());
|
|
|
|
|
|
|
|
let mut file = File::open(pathbuf.to_str().unwrap()).unwrap();
|
|
|
|
let mut program_data = Vec::new();
|
|
|
|
file.read_to_end(&mut program_data).unwrap();
|
|
|
|
let max_len = program_data.len();
|
|
|
|
let minimum_balance_for_buffer = rpc_client
|
|
|
|
.get_minimum_balance_for_rent_exemption(
|
|
|
|
UpgradeableLoaderState::programdata_len(max_len).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let mut config = CliConfig::recent_for_tests();
|
|
|
|
let keypair = Keypair::new();
|
|
|
|
config.json_rpc_url = test_validator.rpc_url();
|
|
|
|
config.signers = vec![&keypair];
|
|
|
|
config.command = CliCommand::Airdrop {
|
|
|
|
faucet_host: None,
|
|
|
|
faucet_port: faucet_addr.port(),
|
|
|
|
pubkey: None,
|
|
|
|
lamports: 100 * minimum_balance_for_buffer,
|
|
|
|
};
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
|
|
|
|
// Write a buffer
|
|
|
|
let buffer_keypair = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::WriteBuffer {
|
|
|
|
program_location: pathbuf.to_str().unwrap().to_string(),
|
|
|
|
buffer_signer_index: Some(1),
|
|
|
|
buffer_pubkey: Some(buffer_keypair.pubkey()),
|
|
|
|
buffer_authority_signer_index: None,
|
|
|
|
is_final: false,
|
|
|
|
max_len: None,
|
|
|
|
});
|
|
|
|
process_command(&config).unwrap();
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(keypair.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set new authority
|
|
|
|
let new_buffer_authority = Keypair::new();
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::SetBufferAuthority {
|
|
|
|
buffer_pubkey: buffer_keypair.pubkey(),
|
|
|
|
buffer_authority_index: Some(0),
|
|
|
|
new_buffer_authority: Some(new_buffer_authority.pubkey()),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let new_buffer_authority_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
Pubkey::from_str(&new_buffer_authority_str).unwrap(),
|
|
|
|
new_buffer_authority.pubkey()
|
|
|
|
);
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(new_buffer_authority.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set authority to buffer
|
|
|
|
config.signers = vec![&keypair, &new_buffer_authority];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::SetBufferAuthority {
|
|
|
|
buffer_pubkey: buffer_keypair.pubkey(),
|
|
|
|
buffer_authority_index: Some(1),
|
|
|
|
new_buffer_authority: Some(buffer_keypair.pubkey()),
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_authority_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
Pubkey::from_str(&buffer_authority_str).unwrap(),
|
|
|
|
buffer_keypair.pubkey()
|
|
|
|
);
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, Some(buffer_keypair.pubkey()));
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set authority to None
|
|
|
|
config.signers = vec![&keypair, &buffer_keypair];
|
|
|
|
config.command = CliCommand::Program(ProgramCliCommand::SetBufferAuthority {
|
|
|
|
buffer_pubkey: buffer_keypair.pubkey(),
|
|
|
|
buffer_authority_index: Some(1),
|
|
|
|
new_buffer_authority: None,
|
|
|
|
});
|
|
|
|
let response = process_command(&config);
|
|
|
|
let json: Value = serde_json::from_str(&response.unwrap()).unwrap();
|
|
|
|
let buffer_authority_str = json
|
|
|
|
.as_object()
|
|
|
|
.unwrap()
|
|
|
|
.get("Authority")
|
|
|
|
.unwrap()
|
|
|
|
.as_str()
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(buffer_authority_str, "None");
|
|
|
|
let buffer_account = rpc_client
|
|
|
|
.get_account_with_commitment(&buffer_keypair.pubkey(), CommitmentConfig::recent())
|
|
|
|
.unwrap()
|
|
|
|
.value
|
|
|
|
.unwrap();
|
|
|
|
if let UpgradeableLoaderState::Buffer { authority_address } = buffer_account.state().unwrap() {
|
|
|
|
assert_eq!(authority_address, None);
|
|
|
|
} else {
|
|
|
|
panic!("not a buffer account");
|
|
|
|
}
|
2020-12-14 15:35:10 -08:00
|
|
|
}
|