Files
solana/core/tests/rpc.rs

298 lines
9.7 KiB
Rust
Raw Normal View History

use bincode::serialize;
use jsonrpc_core::futures::StreamExt;
use jsonrpc_core_client::transports::ws;
use log::*;
use reqwest::{self, header::CONTENT_TYPE};
use serde_json::{json, Value};
use solana_account_decoder::UiAccount;
use solana_client::{
2020-11-25 17:00:47 -08:00
rpc_client::RpcClient,
rpc_response::{Response, RpcSignatureResult},
};
2020-09-18 22:21:44 -07:00
use solana_core::{rpc_pubsub::gen_client::Client as PubsubClient, test_validator::TestValidator};
use solana_sdk::{
commitment_config::CommitmentConfig,
hash::Hash,
signature::{Keypair, Signer},
system_transaction,
2020-10-19 12:23:14 -07:00
transaction::Transaction,
};
use std::{
collections::HashSet,
net::UdpSocket,
sync::mpsc::channel,
thread::sleep,
time::{Duration, Instant},
};
use tokio::runtime::Runtime;
2020-04-09 18:05:56 -07:00
macro_rules! json_req {
($method: expr, $params: expr) => {{
json!({
"jsonrpc": "2.0",
"id": 1,
"method": $method,
"params": $params,
})
}}
}
2020-11-25 17:00:47 -08:00
fn post_rpc(request: Value, rpc_url: &str) -> Value {
2020-04-09 18:05:56 -07:00
let client = reqwest::blocking::Client::new();
let response = client
2020-11-25 17:00:47 -08:00
.post(rpc_url)
2020-04-09 18:05:56 -07:00
.header(CONTENT_TYPE, "application/json")
.body(request.to_string())
.send()
.unwrap();
serde_json::from_str(&response.text().unwrap()).unwrap()
}
#[test]
fn test_rpc_send_tx() {
solana_logger::setup();
let alice = Keypair::new();
let test_validator = TestValidator::with_no_fees(alice.pubkey());
2020-11-25 17:00:47 -08:00
let rpc_url = test_validator.rpc_url();
let bob_pubkey = solana_sdk::pubkey::new_rand();
2020-04-09 18:05:56 -07:00
let req = json_req!("getRecentBlockhash", json!([]));
2020-11-25 17:00:47 -08:00
let json = post_rpc(req, &rpc_url);
2020-04-09 18:05:56 -07:00
let blockhash: Hash = json["result"]["value"]["blockhash"]
.as_str()
.unwrap()
.parse()
.unwrap();
2019-03-02 10:25:16 -08:00
info!("blockhash: {:?}", blockhash);
let tx = system_transaction::transfer(&alice, &bob_pubkey, 20, blockhash);
let serialized_encoded_tx = bs58::encode(serialize(&tx).unwrap()).into_string();
2020-04-09 18:05:56 -07:00
let req = json_req!("sendTransaction", json!([serialized_encoded_tx]));
2020-11-25 17:00:47 -08:00
let json: Value = post_rpc(req, &rpc_url);
2020-04-09 18:05:56 -07:00
let signature = &json["result"];
let mut confirmed_tx = false;
2020-04-09 18:05:56 -07:00
let request = json_req!("confirmTransaction", [signature]);
2019-09-06 14:30:56 -07:00
for _ in 0..solana_sdk::clock::DEFAULT_TICKS_PER_SLOT {
2020-11-25 17:00:47 -08:00
let json = post_rpc(request.clone(), &rpc_url);
if true == json["result"]["value"] {
confirmed_tx = true;
break;
}
sleep(Duration::from_millis(500));
}
assert_eq!(confirmed_tx, true);
use solana_account_decoder::UiAccountEncoding;
use solana_client::rpc_config::RpcAccountInfoConfig;
let config = RpcAccountInfoConfig {
encoding: Some(UiAccountEncoding::Base64),
commitment: None,
data_slice: None,
};
let req = json_req!(
"getAccountInfo",
json!([bs58::encode(bob_pubkey).into_string(), config])
);
2020-11-25 17:00:47 -08:00
let json: Value = post_rpc(req, &rpc_url);
info!("{:?}", json["result"]["value"]);
}
#[test]
fn test_rpc_invalid_requests() {
solana_logger::setup();
let alice = Keypair::new();
let test_validator = TestValidator::with_no_fees(alice.pubkey());
2020-11-25 17:00:47 -08:00
let rpc_url = test_validator.rpc_url();
let bob_pubkey = solana_sdk::pubkey::new_rand();
// test invalid get_balance request
2020-04-09 18:05:56 -07:00
let req = json_req!("getBalance", json!(["invalid9999"]));
2020-11-25 17:00:47 -08:00
let json = post_rpc(req, &rpc_url);
2020-04-09 18:05:56 -07:00
let the_error = json["error"]["message"].as_str().unwrap();
assert_eq!(the_error, "Invalid param: Invalid");
// test invalid get_account_info request
2020-04-09 18:05:56 -07:00
let req = json_req!("getAccountInfo", json!(["invalid9999"]));
2020-11-25 17:00:47 -08:00
let json = post_rpc(req, &rpc_url);
2020-04-09 18:05:56 -07:00
let the_error = json["error"]["message"].as_str().unwrap();
assert_eq!(the_error, "Invalid param: Invalid");
// test invalid get_account_info request
2020-04-09 18:05:56 -07:00
let req = json_req!("getAccountInfo", json!([bob_pubkey.to_string()]));
2020-11-25 17:00:47 -08:00
let json = post_rpc(req, &rpc_url);
2020-04-09 18:05:56 -07:00
let the_value = &json["result"]["value"];
assert!(the_value.is_null());
}
#[test]
fn test_rpc_subscriptions() {
solana_logger::setup();
let alice = Keypair::new();
let test_validator = TestValidator::with_no_fees(alice.pubkey());
let transactions_socket = UdpSocket::bind("0.0.0.0:0").unwrap();
2020-11-25 17:00:47 -08:00
transactions_socket.connect(test_validator.tpu()).unwrap();
let rpc_client = RpcClient::new(test_validator.rpc_url());
let recent_blockhash = rpc_client.get_recent_blockhash().unwrap().0;
// Create transaction signatures to subscribe to
let transactions: Vec<Transaction> = (0..1000)
2020-10-19 12:23:14 -07:00
.map(|_| {
2020-11-25 17:00:47 -08:00
system_transaction::transfer(
&alice,
&solana_sdk::pubkey::new_rand(),
1,
recent_blockhash,
2020-11-25 17:00:47 -08:00
)
2020-10-19 12:23:14 -07:00
})
.collect();
let mut signature_set: HashSet<String> = transactions
.iter()
.map(|tx| tx.signatures[0].to_string())
.collect();
let account_set: HashSet<String> = transactions
.iter()
.map(|tx| tx.message.account_keys[1].to_string())
.collect();
// Track when subscriptions are ready
let (ready_sender, ready_receiver) = channel::<()>();
// Track account notifications are received
let (account_sender, account_receiver) = channel::<Response<UiAccount>>();
// Track when status notifications are received
let (status_sender, status_receiver) = channel::<(String, Response<RpcSignatureResult>)>();
// Create the pub sub runtime
let rt = Runtime::new().unwrap();
let rpc_pubsub_url = test_validator.rpc_pubsub_url();
let signature_set_clone = signature_set.clone();
rt.spawn(async move {
let connect = ws::try_connect::<PubsubClient>(&rpc_pubsub_url).unwrap();
let client = connect.await.unwrap();
// Subscribe to signature notifications
for sig in signature_set_clone {
let status_sender = status_sender.clone();
let mut sig_sub = client
.signature_subscribe(sig.clone(), None)
.unwrap_or_else(|err| panic!("sig sub err: {:#?}", err));
tokio::spawn(async move {
let response = sig_sub.next().await.unwrap();
status_sender
.send((sig.clone(), response.unwrap()))
.unwrap();
});
}
// Subscribe to account notifications
for pubkey in account_set {
let account_sender = account_sender.clone();
let mut client_sub = client
.account_subscribe(pubkey, None)
.unwrap_or_else(|err| panic!("acct sub err: {:#?}", err));
tokio::spawn(async move {
let response = client_sub.next().await.unwrap();
account_sender.send(response.unwrap()).unwrap();
});
}
// Signal ready after the next slot notification
let mut slot_sub = client
.slot_subscribe()
.unwrap_or_else(|err| panic!("sig sub err: {:#?}", err));
tokio::spawn(async move {
let _response = slot_sub.next().await.unwrap();
ready_sender.send(()).unwrap();
});
});
// Wait for signature subscriptions
ready_receiver.recv_timeout(Duration::from_secs(2)).unwrap();
2020-11-25 17:00:47 -08:00
let rpc_client = RpcClient::new(test_validator.rpc_url());
let mut mint_balance = rpc_client
.get_balance_with_commitment(&alice.pubkey(), CommitmentConfig::processed())
.unwrap()
.value;
assert!(mint_balance >= transactions.len() as u64);
// Send all transactions to tpu socket for processing
transactions.iter().for_each(|tx| {
transactions_socket
.send(&bincode::serialize(&tx).unwrap())
.unwrap();
});
// Track mint balance to know when transactions have completed
let now = Instant::now();
let expected_mint_balance = mint_balance - transactions.len() as u64;
while mint_balance != expected_mint_balance && now.elapsed() < Duration::from_secs(5) {
mint_balance = rpc_client
.get_balance_with_commitment(&alice.pubkey(), CommitmentConfig::processed())
.unwrap()
.value;
sleep(Duration::from_millis(100));
}
// Wait for all signature subscriptions
let deadline = Instant::now() + Duration::from_secs(7);
while !signature_set.is_empty() {
let timeout = deadline.saturating_duration_since(Instant::now());
match status_receiver.recv_timeout(timeout) {
Ok((sig, result)) => {
if let RpcSignatureResult::ProcessedSignature(result) = result.value {
assert!(result.err.is_none());
assert!(signature_set.remove(&sig));
} else {
panic!("Unexpected result");
}
}
Err(_err) => {
2020-12-13 17:26:34 -08:00
panic!(
"recv_timeout, {}/{} signatures remaining",
signature_set.len(),
transactions.len()
);
}
}
}
let deadline = Instant::now() + Duration::from_secs(5);
let mut account_notifications = transactions.len();
while account_notifications > 0 {
let timeout = deadline.saturating_duration_since(Instant::now());
match account_receiver.recv_timeout(timeout) {
Ok(result) => {
assert_eq!(result.value.lamports, 1);
account_notifications -= 1;
}
Err(_err) => {
2020-12-13 17:26:34 -08:00
panic!(
"recv_timeout, {}/{} accounts remaining",
account_notifications,
transactions.len()
);
}
}
}
}