Wrap all exceptions and panics to relay them over JNI

This commit is contained in:
Jack Grigg 2019-02-12 02:56:17 +00:00
parent 25e79f042b
commit 933fb07f42
No known key found for this signature in database
GPG Key ID: 1B8D649257DB0829
2 changed files with 154 additions and 170 deletions

View File

@ -7,13 +7,16 @@ mod utils;
const SAPLING_CONSENSUS_BRANCH_ID: u32 = 0x76b8_09bb; const SAPLING_CONSENSUS_BRANCH_ID: u32 = 0x76b8_09bb;
use android_logger::Filter; use android_logger::Filter;
use failure::format_err;
use jni::{ use jni::{
objects::{JClass, JString}, objects::{JClass, JString},
sys::{jboolean, jbyteArray, jint, jlong, jobjectArray, jstring, JNI_FALSE, JNI_TRUE}, sys::{jboolean, jbyteArray, jint, jlong, jobjectArray, jstring, JNI_FALSE, JNI_TRUE},
JNIEnv, JNIEnv,
}; };
use log::Level; use log::Level;
use std::panic;
use std::path::Path; use std::path::Path;
use std::ptr;
use zcash_client_backend::{ use zcash_client_backend::{
constants::{HRP_SAPLING_EXTENDED_SPENDING_KEY_TEST, HRP_SAPLING_PAYMENT_ADDRESS_TEST}, constants::{HRP_SAPLING_EXTENDED_SPENDING_KEY_TEST, HRP_SAPLING_PAYMENT_ADDRESS_TEST},
encoding::{ encoding::{
@ -26,9 +29,12 @@ use zcash_client_backend::{
use zcash_primitives::transaction::components::Amount; use zcash_primitives::transaction::components::Amount;
use zip32::ExtendedFullViewingKey; use zip32::ExtendedFullViewingKey;
use crate::sql::{ use crate::{
sql::{
get_address, get_balance, init_accounts_table, init_blocks_table, init_data_database, get_address, get_balance, init_accounts_table, init_blocks_table, init_data_database,
scan_cached_blocks, send_to_address, scan_cached_blocks, send_to_address,
},
utils::exception::unwrap_exc_or,
}; };
#[no_mangle] #[no_mangle]
@ -52,15 +58,14 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_initDataDb(
_: JClass<'_>, _: JClass<'_>,
db_data: JString<'_>, db_data: JString<'_>,
) -> jboolean { ) -> jboolean {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
match init_data_database(&db_data) { init_data_database(&db_data)
Ok(()) => JNI_TRUE, .map(|()| JNI_TRUE)
Err(e) => { .map_err(|e| format_err!("Error while initializing data DB: {}", e))
error!("Error while initializing data DB: {}", e); });
JNI_FALSE unwrap_exc_or(&env, res, JNI_FALSE)
}
}
} }
#[no_mangle] #[no_mangle]
@ -71,11 +76,16 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_initAccountsTab
seed: jbyteArray, seed: jbyteArray,
accounts: jint, accounts: jint,
) -> jobjectArray { ) -> jobjectArray {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let seed = env.convert_byte_array(seed).unwrap(); let seed = env.convert_byte_array(seed).unwrap();
let accounts = if accounts >= 0 {
accounts as u32
} else {
return Err(format_err!("accounts argument must be positive"));
};
let ret = if accounts >= 0 { let extsks: Vec<_> = (0..accounts)
let extsks: Vec<_> = (0..accounts as u32)
.map(|account| spending_key(&seed, 1, account)) .map(|account| spending_key(&seed, 1, account))
.collect(); .collect();
let extfvks: Vec<_> = extsks.iter().map(ExtendedFullViewingKey::from).collect(); let extfvks: Vec<_> = extsks.iter().map(ExtendedFullViewingKey::from).collect();
@ -83,23 +93,9 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_initAccountsTab
match init_accounts_table(&db_data, &extfvks) { match init_accounts_table(&db_data, &extfvks) {
Ok(()) => { Ok(()) => {
// Return the ExtendedSpendingKeys for the created accounts // Return the ExtendedSpendingKeys for the created accounts
extsks Ok(utils::rust_vec_to_java(
}
Err(e) => {
error!("Error while initializing accounts: {}", e);
// Return an empty array to indicate an error
vec![]
}
}
} else {
error!("accounts argument must be positive");
// Return an empty array to indicate an error
vec![]
};
utils::rust_vec_to_java(
&env, &env,
ret, extsks,
"java/lang/String", "java/lang/String",
|env, extsk| { |env, extsk| {
env.new_string(encode_extended_spending_key( env.new_string(encode_extended_spending_key(
@ -108,7 +104,12 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_initAccountsTab
)) ))
}, },
|env| env.new_string(""), |env| env.new_string(""),
) ))
}
Err(e) => Err(format_err!("Error while initializing accounts: {}", e)),
}
});
unwrap_exc_or(&env, res, ptr::null_mut())
} }
#[no_mangle] #[no_mangle]
@ -120,22 +121,21 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_initBlocksTable
time: jlong, time: jlong,
sapling_tree: jbyteArray, sapling_tree: jbyteArray,
) -> jboolean { ) -> jboolean {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let time = if time >= 0 && time <= jlong::from(u32::max_value()) { let time = if time >= 0 && time <= jlong::from(u32::max_value()) {
time as u32 time as u32
} else { } else {
error!("time argument must fit in a u32"); return Err(format_err!("time argument must fit in a u32"));
return JNI_FALSE;
}; };
let sapling_tree = env.convert_byte_array(sapling_tree).unwrap(); let sapling_tree = env.convert_byte_array(sapling_tree).unwrap();
match init_blocks_table(&db_data, height, time, &sapling_tree) { match init_blocks_table(&db_data, height, time, &sapling_tree) {
Ok(()) => JNI_TRUE, Ok(()) => Ok(JNI_TRUE),
Err(e) => { Err(e) => Err(format_err!("Error while initializing blocks table: {}", e)),
error!("Error while initializing data DB: {}", e);
JNI_FALSE
}
} }
});
unwrap_exc_or(&env, res, JNI_FALSE)
} }
#[no_mangle] #[no_mangle]
@ -145,26 +145,23 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_getAddress(
db_data: JString<'_>, db_data: JString<'_>,
account: jint, account: jint,
) -> jstring { ) -> jstring {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let account = if account >= 0 {
let addr = match account { account as u32
acc if acc >= 0 => match get_address(&db_data, acc as u32) { } else {
Ok(addr) => addr, return Err(format_err!("account argument must be positive"));
Err(e) => {
error!("Error while fetching address: {}", e);
// Return an empty string to indicate an error
String::default()
}
},
_ => {
error!("account argument must be positive");
// Return an empty string to indicate an error
String::default()
}
}; };
match get_address(&db_data, account) {
Ok(addr) => {
let output = env.new_string(addr).expect("Couldn't create Java string!"); let output = env.new_string(addr).expect("Couldn't create Java string!");
output.into_inner() Ok(output.into_inner())
}
Err(e) => Err(format_err!("Error while fetching address: {}", e)),
}
});
unwrap_exc_or(&env, res, ptr::null_mut())
} }
#[no_mangle] #[no_mangle]
@ -174,21 +171,20 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_getBalance(
db_data: JString<'_>, db_data: JString<'_>,
account: jint, account: jint,
) -> jlong { ) -> jlong {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let account = if account >= 0 { let account = if account >= 0 {
account as u32 account as u32
} else { } else {
error!("account argument must be positive"); return Err(format_err!("account argument must be positive"));
return -1;
}; };
match get_balance(&db_data, account) { match get_balance(&db_data, account) {
Ok(balance) => balance.0, Ok(balance) => Ok(balance.0),
Err(e) => { Err(e) => Err(format_err!("Error while fetching balance: {}", e)),
error!("Error while fetching balance: {}", e);
-1
}
} }
});
unwrap_exc_or(&env, res, -1)
} }
#[no_mangle] #[no_mangle]
@ -198,19 +194,18 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_getReceivedMemo
db_data: JString<'_>, db_data: JString<'_>,
id_note: jlong, id_note: jlong,
) -> jstring { ) -> jstring {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let memo = match crate::sql::get_received_memo_as_utf8(db_data, id_note) { let memo = match crate::sql::get_received_memo_as_utf8(db_data, id_note) {
Ok(memo) => memo.unwrap_or_default(), Ok(memo) => memo.unwrap_or_default(),
Err(e) => { Err(e) => return Err(format_err!("Error while fetching memo: {}", e)),
error!("Error while fetching memo: {}", e);
// Return an empty string to indicate an error
String::default()
}
}; };
let output = env.new_string(memo).expect("Couldn't create Java string!"); let output = env.new_string(memo).expect("Couldn't create Java string!");
output.into_inner() Ok(output.into_inner())
});
unwrap_exc_or(&env, res, ptr::null_mut())
} }
#[no_mangle] #[no_mangle]
@ -220,19 +215,18 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_getSentMemoAsUt
db_data: JString<'_>, db_data: JString<'_>,
id_note: jlong, id_note: jlong,
) -> jstring { ) -> jstring {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let memo = match crate::sql::get_sent_memo_as_utf8(db_data, id_note) { let memo = match crate::sql::get_sent_memo_as_utf8(db_data, id_note) {
Ok(memo) => memo.unwrap_or_default(), Ok(memo) => memo.unwrap_or_default(),
Err(e) => { Err(e) => return Err(format_err!("Error while fetching memo: {}", e)),
error!("Error while fetching memo: {}", e);
// Return an empty string to indicate an error
String::default()
}
}; };
let output = env.new_string(memo).expect("Couldn't create Java string!"); let output = env.new_string(memo).expect("Couldn't create Java string!");
output.into_inner() Ok(output.into_inner())
});
unwrap_exc_or(&env, res, ptr::null_mut())
} }
#[no_mangle] #[no_mangle]
@ -242,16 +236,16 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_scanBlocks(
db_cache: JString<'_>, db_cache: JString<'_>,
db_data: JString<'_>, db_data: JString<'_>,
) -> jboolean { ) -> jboolean {
let res = panic::catch_unwind(|| {
let db_cache = utils::java_string_to_rust(&env, db_cache); let db_cache = utils::java_string_to_rust(&env, db_cache);
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
match scan_cached_blocks(&db_cache, &db_data) { match scan_cached_blocks(&db_cache, &db_data) {
Ok(()) => JNI_TRUE, Ok(()) => Ok(JNI_TRUE),
Err(e) => { Err(e) => Err(format_err!("Error while scanning blocks: {}", e)),
error!("Error while scanning blocks: {}", e);
JNI_FALSE
}
} }
});
unwrap_exc_or(&env, res, JNI_FALSE)
} }
#[no_mangle] #[no_mangle]
@ -267,12 +261,12 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_sendToAddress(
spend_params: JString<'_>, spend_params: JString<'_>,
output_params: JString<'_>, output_params: JString<'_>,
) -> jlong { ) -> jlong {
let res = panic::catch_unwind(|| {
let db_data = utils::java_string_to_rust(&env, db_data); let db_data = utils::java_string_to_rust(&env, db_data);
let account = if account >= 0 { let account = if account >= 0 {
account as u32 account as u32
} else { } else {
error!("account argument must be positive"); return Err(format_err!("account argument must be positive"));
return -1;
}; };
let extsk = utils::java_string_to_rust(&env, extsk); let extsk = utils::java_string_to_rust(&env, extsk);
let to = utils::java_string_to_rust(&env, to); let to = utils::java_string_to_rust(&env, to);
@ -281,29 +275,22 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_sendToAddress(
let spend_params = utils::java_string_to_rust(&env, spend_params); let spend_params = utils::java_string_to_rust(&env, spend_params);
let output_params = utils::java_string_to_rust(&env, output_params); let output_params = utils::java_string_to_rust(&env, output_params);
let extsk = match decode_extended_spending_key(HRP_SAPLING_EXTENDED_SPENDING_KEY_TEST, &extsk) { let extsk =
match decode_extended_spending_key(HRP_SAPLING_EXTENDED_SPENDING_KEY_TEST, &extsk) {
Ok(extsk) => extsk, Ok(extsk) => extsk,
Err(e) => { Err(e) => {
error!("Invalid ExtendedSpendingKey: {}", e); return Err(format_err!("Invalid ExtendedSpendingKey: {}", e));
return -1;
} }
}; };
let to = match decode_payment_address(HRP_SAPLING_PAYMENT_ADDRESS_TEST, &to) { let to = match decode_payment_address(HRP_SAPLING_PAYMENT_ADDRESS_TEST, &to) {
Ok(to) => to, Ok(to) => to,
Err(e) => { Err(e) => {
error!("Invalid PaymentAddress: {}", e); return Err(format_err!("Invalid PaymentAddress: {}", e));
return -1;
} }
}; };
let memo = match Memo::from_str(&memo) { let memo = Some(Memo::from_str(&memo)?);
Ok(memo) => Some(memo),
Err(e) => {
error!("{}", e);
return -1;
}
};
let prover = LocalTxProver::new( let prover = LocalTxProver::new(
Path::new(&spend_params), Path::new(&spend_params),
@ -312,7 +299,7 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_sendToAddress(
"657e3d38dbb5cb5e7dd2970e8b03d69b4787dd907285b5a7f0790dcc8072f60bf593b32cc2d1c030e00ff5ae64bf84c5c3beb84ddc841d48264b4a171744d028", "657e3d38dbb5cb5e7dd2970e8b03d69b4787dd907285b5a7f0790dcc8072f60bf593b32cc2d1c030e00ff5ae64bf84c5c3beb84ddc841d48264b4a171744d028",
); );
match send_to_address( send_to_address(
&db_data, &db_data,
SAPLING_CONSENSUS_BRANCH_ID, SAPLING_CONSENSUS_BRANCH_ID,
prover, prover,
@ -320,11 +307,8 @@ pub unsafe extern "C" fn Java_cash_z_wallet_sdk_jni_JniConverter_sendToAddress(
&to, &to,
value, value,
memo, memo,
) { )
Ok(tx_row) => tx_row, .map_err(|e| format_err!("Error while sending funds: {}", e))
Err(e) => { });
error!("Error while sending funds: {}", e); unwrap_exc_or(&env, res, -1)
-1
}
}
} }

View File

@ -33,7 +33,7 @@ fn address_from_extfvk(extfvk: &ExtendedFullViewingKey) -> String {
encode_payment_address(HRP_SAPLING_PAYMENT_ADDRESS_TEST, &addr) encode_payment_address(HRP_SAPLING_PAYMENT_ADDRESS_TEST, &addr)
} }
pub fn init_cache_database<P: AsRef<Path>>(db_cache: P) -> rusqlite::Result<()> { pub fn init_cache_database<P: AsRef<Path>>(db_cache: P) -> Result<(), Error> {
let cache = Connection::open(db_cache)?; let cache = Connection::open(db_cache)?;
cache.execute( cache.execute(
"CREATE TABLE IF NOT EXISTS compactblocks ( "CREATE TABLE IF NOT EXISTS compactblocks (
@ -45,7 +45,7 @@ pub fn init_cache_database<P: AsRef<Path>>(db_cache: P) -> rusqlite::Result<()>
Ok(()) Ok(())
} }
pub fn init_data_database<P: AsRef<Path>>(db_data: P) -> rusqlite::Result<()> { pub fn init_data_database<P: AsRef<Path>>(db_data: P) -> Result<(), Error> {
let data = Connection::open(db_data)?; let data = Connection::open(db_data)?;
data.execute( data.execute(
"CREATE TABLE IF NOT EXISTS accounts ( "CREATE TABLE IF NOT EXISTS accounts (