mod error;
mod serialize;
mod serves;
pub use self::error::Error;
use super::core;
use super::keystore::KdfDepthLevel;
use super::storage::{self, StorageController};
use super::util::{align_bytes, to_arr, to_chain_id, to_even_str, to_u64, trim_hex, ToHex};
use hdwallet::WManager;
use jsonrpc_core::{Error as JsonRpcError, IoHandler, Params};
use jsonrpc_http_server::{AccessControlAllowOrigin, DomainsValidation, ServerBuilder};
use log::LogLevel;
use serde::de::DeserializeOwned;
use serde::Serialize;
use serde_json::{self, Value};
use std::cell::RefCell;
use std::net::SocketAddr;
use std::sync::{Arc, Mutex};
fn wrapper<T: Serialize>(value: Result<T, Error>) -> Result<Value, JsonRpcError> {
if value.is_err() {
return Err(JsonRpcError::invalid_params(
value.err().unwrap().to_string(),
));
}
let value = value.unwrap();
let result = serde_json::to_value(value);
match result {
Ok(value) => Ok(value),
Err(e) => Err(JsonRpcError::invalid_params(e.to_string())),
}
}
fn parse<T>(p: Params) -> Result<T, JsonRpcError>
where
T: DeserializeOwned,
{
p.parse()
.map_err(|_| JsonRpcError::invalid_params("Corrupted input parameters".to_string()))
}
pub fn start(
addr: &SocketAddr,
chain_name: &str,
storage_ctrl: Arc<Box<StorageController>>,
sec_level: Option<KdfDepthLevel>,
) {
let sec_level = sec_level.unwrap_or_default();
let storage_ctrl = Arc::new(Mutex::new(storage_ctrl));
let chain_id = match to_chain_id(chain_name) {
Some(id) => id,
None => panic!(format!("Invalid chain name: {}", chain_name)),
};
let wallet_manager = match WManager::new(None) {
Ok(wm) => Arc::new(Mutex::new(RefCell::new(wm))),
Err(e) => panic!("Can't create HID endpoint: {}", e.to_string()),
};
let mut io = IoHandler::default();
{
io.add_method("emerald_currentVersion", move |p: Params| {
wrapper(serves::current_version(parse(p)?))
});
}
{
io.add_method("emerald_heartbeat", move |p: Params| {
wrapper(serves::heartbeat(parse(p)?))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_listAddresses", move |p: Params| {
wrapper(serves::list_addresses(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_importAddress", move |p: Params| {
wrapper(serves::import_address(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_deleteAddress", move |p: Params| {
wrapper(serves::delete_address(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_listAccounts", move |p: Params| {
wrapper(serves::list_accounts(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_hideAccount", move |p: Params| {
wrapper(serves::hide_account(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_unhideAccount", move |p: Params| {
wrapper(serves::unhide_account(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_shakeAccount", move |p: Params| {
wrapper(serves::shake_account(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_updateAccount", move |p: Params| {
wrapper(serves::update_account(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_importAccount", move |p: Params| {
wrapper(serves::import_account(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_exportAccount", move |p: Params| {
wrapper(serves::export_account(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_newAccount", move |p: Params| {
wrapper(serves::new_account(parse(p)?, &sec_level, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
let wm = Arc::clone(&wallet_manager);
io.add_method("emerald_signTransaction", move |p: Params| {
wrapper(serves::sign_transaction(
parse(p)?,
&storage_ctrl,
chain_id,
&wm,
))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
let wm = Arc::clone(&wallet_manager);
io.add_method("emerald_sign", move |p: Params| {
wrapper(serves::sign(parse(p)?, &storage_ctrl, &wm))
});
}
{
io.add_method("emerald_encodeFunctionCall", move |p: Params| {
wrapper(serves::encode_function_call(parse(p)?))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_listContracts", move |p: Params| {
wrapper(serves::list_contracts(parse(p)?, &storage_ctrl))
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_importContract", move |p: Params| {
wrapper(serves::import_contract(parse(p)?, &storage_ctrl))
});
}
{
io.add_method("emerald_generateMnemonic", move |_: Params| {
wrapper(serves::generate_mnemonic())
});
}
{
let storage_ctrl = Arc::clone(&storage_ctrl);
io.add_method("emerald_importMnemonic", move |p: Params| {
wrapper(serves::import_mnemonic(
parse(p)?,
&sec_level,
&storage_ctrl,
))
});
}
let server = ServerBuilder::new(io)
.cors(DomainsValidation::AllowOnly(vec![
AccessControlAllowOrigin::Any,
AccessControlAllowOrigin::Null,
]))
.start_http(addr)
.expect("Expect to build HTTP RPC server");
if log_enabled!(LogLevel::Info) {
info!("Connector started on http://{}", server.address());
}
server.wait();
}