mirror of
https://github.com/mimblewimble/grin.git
synced 2025-01-21 03:21:08 +03:00
Error handling using failure in API (#949)
This PR adresses #166 Error handling in wallet was ported to failure in https://github.com/mimblewimble/grin/pull/713 Using the same error model makes wallet code simpler and may simplify migration to Hyper.
This commit is contained in:
parent
ffa5bfe16f
commit
b28de95da4
10 changed files with 171 additions and 117 deletions
2
Cargo.lock
generated
2
Cargo.lock
generated
|
@ -630,6 +630,8 @@ dependencies = [
|
||||||
name = "grin_api"
|
name = "grin_api"
|
||||||
version = "0.2.0"
|
version = "0.2.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"failure 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"failure_derive 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"grin_chain 0.2.0",
|
"grin_chain 0.2.0",
|
||||||
"grin_core 0.2.0",
|
"grin_core 0.2.0",
|
||||||
"grin_p2p 0.2.0",
|
"grin_p2p 0.2.0",
|
||||||
|
|
|
@ -6,6 +6,8 @@ workspace = ".."
|
||||||
publish = false
|
publish = false
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
failure = "0.1"
|
||||||
|
failure_derive = "0.1"
|
||||||
hyper = "0.10"
|
hyper = "0.10"
|
||||||
iron = "0.5"
|
iron = "0.5"
|
||||||
lazy_static = "0.2"
|
lazy_static = "0.2"
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
|
|
||||||
//! High level JSON/HTTP client API
|
//! High level JSON/HTTP client API
|
||||||
|
|
||||||
|
use failure::{Fail, ResultExt};
|
||||||
use hyper;
|
use hyper;
|
||||||
use hyper::client::Response;
|
use hyper::client::Response;
|
||||||
use hyper::status::{StatusClass, StatusCode};
|
use hyper::status::{StatusClass, StatusCode};
|
||||||
|
@ -21,7 +22,7 @@ use serde::{Deserialize, Serialize};
|
||||||
use serde_json;
|
use serde_json;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
|
|
||||||
use rest::Error;
|
use rest::{Error, ErrorKind};
|
||||||
|
|
||||||
/// Helper function to easily issue a HTTP GET request against a given URL that
|
/// Helper function to easily issue a HTTP GET request against a given URL that
|
||||||
/// returns a JSON object. Handles request building, JSON deserialization and
|
/// returns a JSON object. Handles request building, JSON deserialization and
|
||||||
|
@ -32,8 +33,11 @@ where
|
||||||
{
|
{
|
||||||
let client = hyper::Client::new();
|
let client = hyper::Client::new();
|
||||||
let res = check_error(client.get(url).send())?;
|
let res = check_error(client.get(url).send())?;
|
||||||
serde_json::from_reader(res)
|
serde_json::from_reader(res).map_err(|e| {
|
||||||
.map_err(|e| Error::Internal(format!("Server returned invalid JSON: {}", e)))
|
e.context(ErrorKind::Internal(
|
||||||
|
"Server returned invalid JSON".to_owned(),
|
||||||
|
)).into()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Helper function to easily issue a HTTP POST request with the provided JSON
|
/// Helper function to easily issue a HTTP POST request with the provided JSON
|
||||||
|
@ -44,8 +48,9 @@ pub fn post<'a, IN>(url: &'a str, input: &IN) -> Result<(), Error>
|
||||||
where
|
where
|
||||||
IN: Serialize,
|
IN: Serialize,
|
||||||
{
|
{
|
||||||
let in_json = serde_json::to_string(input)
|
let in_json = serde_json::to_string(input).context(ErrorKind::Internal(
|
||||||
.map_err(|e| Error::Internal(format!("Could not serialize data to JSON: {}", e)))?;
|
"Could not serialize data to JSON".to_owned(),
|
||||||
|
))?;
|
||||||
let client = hyper::Client::new();
|
let client = hyper::Client::new();
|
||||||
let _res = check_error(client.post(url).body(&mut in_json.as_bytes()).send())?;
|
let _res = check_error(client.post(url).body(&mut in_json.as_bytes()).send())?;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -54,24 +59,27 @@ where
|
||||||
// convert hyper error and check for non success response codes
|
// convert hyper error and check for non success response codes
|
||||||
fn check_error(res: hyper::Result<Response>) -> Result<Response, Error> {
|
fn check_error(res: hyper::Result<Response>) -> Result<Response, Error> {
|
||||||
if let Err(e) = res {
|
if let Err(e) = res {
|
||||||
return Err(Error::Internal(format!("Error during request: {}", e)));
|
return Err(
|
||||||
|
e.context(ErrorKind::Internal("Error during request".to_owned()))
|
||||||
|
.into(),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
let mut response = res.unwrap();
|
let mut response = res.unwrap();
|
||||||
match response.status.class() {
|
match response.status.class() {
|
||||||
StatusClass::Success => Ok(response),
|
StatusClass::Success => Ok(response),
|
||||||
StatusClass::ServerError => Err(Error::Internal(format!(
|
StatusClass::ServerError => Err(ErrorKind::Internal(format!(
|
||||||
"Server error: {}",
|
"Server error: {}",
|
||||||
err_msg(&mut response)
|
err_msg(&mut response)
|
||||||
))),
|
)))?,
|
||||||
StatusClass::ClientError => if response.status == StatusCode::NotFound {
|
StatusClass::ClientError => if response.status == StatusCode::NotFound {
|
||||||
Err(Error::NotFound)
|
Err(ErrorKind::NotFound)?
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Argument(format!(
|
Err(ErrorKind::Argument(format!(
|
||||||
"Argument error: {}",
|
"Argument error: {}",
|
||||||
err_msg(&mut response)
|
err_msg(&mut response)
|
||||||
)))
|
)))?
|
||||||
},
|
},
|
||||||
_ => Err(Error::Internal(format!("Unrecognized error."))),
|
_ => Err(ErrorKind::Internal(format!("Unrecognized error.")))?,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,25 +16,26 @@ use std::io::Read;
|
||||||
use std::sync::{Arc, RwLock, Weak};
|
use std::sync::{Arc, RwLock, Weak};
|
||||||
use std::thread;
|
use std::thread;
|
||||||
|
|
||||||
use iron::prelude::*;
|
use failure::{Fail, ResultExt};
|
||||||
use iron::Handler;
|
use iron::Handler;
|
||||||
|
use iron::prelude::*;
|
||||||
use iron::status;
|
use iron::status;
|
||||||
use urlencoded::UrlEncodedQuery;
|
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use serde_json;
|
use serde_json;
|
||||||
|
use urlencoded::UrlEncodedQuery;
|
||||||
|
|
||||||
use chain;
|
use chain;
|
||||||
use core::core::{OutputFeatures, OutputIdentifier, Transaction};
|
|
||||||
use core::core::hash::{Hash, Hashed};
|
use core::core::hash::{Hash, Hashed};
|
||||||
|
use core::core::{OutputFeatures, OutputIdentifier, Transaction};
|
||||||
use core::ser;
|
use core::ser;
|
||||||
use pool;
|
|
||||||
use p2p;
|
use p2p;
|
||||||
|
use pool;
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
use rest::*;
|
use rest::*;
|
||||||
use util::secp::pedersen::Commitment;
|
|
||||||
use types::*;
|
use types::*;
|
||||||
use util;
|
use util;
|
||||||
use util::LOGGER;
|
use util::LOGGER;
|
||||||
|
use util::secp::pedersen::Commitment;
|
||||||
|
|
||||||
// All handlers use `Weak` references instead of `Arc` to avoid cycles that
|
// All handlers use `Weak` references instead of `Arc` to avoid cycles that
|
||||||
// can never be destroyed. These 2 functions are simple helpers to reduce the
|
// can never be destroyed. These 2 functions are simple helpers to reduce the
|
||||||
|
@ -67,8 +68,10 @@ struct OutputHandler {
|
||||||
|
|
||||||
impl OutputHandler {
|
impl OutputHandler {
|
||||||
fn get_output(&self, id: &str) -> Result<Output, Error> {
|
fn get_output(&self, id: &str) -> Result<Output, Error> {
|
||||||
let c = util::from_hex(String::from(id))
|
let c = util::from_hex(String::from(id)).context(ErrorKind::Argument(format!(
|
||||||
.map_err(|_| Error::Argument(format!("Not a valid commitment: {}", id)))?;
|
"Not a valid commitment: {}",
|
||||||
|
id
|
||||||
|
)))?;
|
||||||
let commit = Commitment::from_vec(c);
|
let commit = Commitment::from_vec(c);
|
||||||
|
|
||||||
// We need the features here to be able to generate the necessary hash
|
// We need the features here to be able to generate the necessary hash
|
||||||
|
@ -85,7 +88,7 @@ impl OutputHandler {
|
||||||
return Ok(Output::new(&commit));
|
return Ok(Output::new(&commit));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Err(Error::NotFound)
|
Err(ErrorKind::NotFound)?
|
||||||
}
|
}
|
||||||
|
|
||||||
fn outputs_by_ids(&self, req: &mut Request) -> Vec<Output> {
|
fn outputs_by_ids(&self, req: &mut Request) -> Vec<Output> {
|
||||||
|
@ -499,19 +502,18 @@ impl Handler for ChainCompactHandler {
|
||||||
///
|
///
|
||||||
/// Optionally return results as "compact blocks" by passing "?compact" query param
|
/// Optionally return results as "compact blocks" by passing "?compact" query param
|
||||||
/// GET /v1/blocks/<hash>?compact
|
/// GET /v1/blocks/<hash>?compact
|
||||||
///
|
|
||||||
pub struct BlockHandler {
|
pub struct BlockHandler {
|
||||||
pub chain: Weak<chain::Chain>,
|
pub chain: Weak<chain::Chain>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BlockHandler {
|
impl BlockHandler {
|
||||||
fn get_block(&self, h: &Hash) -> Result<BlockPrintable, Error> {
|
fn get_block(&self, h: &Hash) -> Result<BlockPrintable, Error> {
|
||||||
let block = w(&self.chain).get_block(h).map_err(|_| Error::NotFound)?;
|
let block = w(&self.chain).get_block(h).context(ErrorKind::NotFound)?;
|
||||||
Ok(BlockPrintable::from_block(&block, w(&self.chain), false))
|
Ok(BlockPrintable::from_block(&block, w(&self.chain), false))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_compact_block(&self, h: &Hash) -> Result<CompactBlockPrintable, Error> {
|
fn get_compact_block(&self, h: &Hash) -> Result<CompactBlockPrintable, Error> {
|
||||||
let block = w(&self.chain).get_block(h).map_err(|_| Error::NotFound)?;
|
let block = w(&self.chain).get_block(h).context(ErrorKind::NotFound)?;
|
||||||
Ok(CompactBlockPrintable::from_compact_block(
|
Ok(CompactBlockPrintable::from_compact_block(
|
||||||
&block.as_compact_block(),
|
&block.as_compact_block(),
|
||||||
w(&self.chain),
|
w(&self.chain),
|
||||||
|
@ -523,14 +525,16 @@ impl BlockHandler {
|
||||||
if let Ok(height) = input.parse() {
|
if let Ok(height) = input.parse() {
|
||||||
match w(&self.chain).get_header_by_height(height) {
|
match w(&self.chain).get_header_by_height(height) {
|
||||||
Ok(header) => return Ok(header.hash()),
|
Ok(header) => return Ok(header.hash()),
|
||||||
Err(_) => return Err(Error::NotFound),
|
Err(_) => return Err(ErrorKind::NotFound)?,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
static ref RE: Regex = Regex::new(r"[0-9a-fA-F]{64}").unwrap();
|
static ref RE: Regex = Regex::new(r"[0-9a-fA-F]{64}").unwrap();
|
||||||
}
|
}
|
||||||
if !RE.is_match(&input) {
|
if !RE.is_match(&input) {
|
||||||
return Err(Error::Argument(String::from("Not a valid hash or height.")));
|
return Err(ErrorKind::Argument(
|
||||||
|
"Not a valid hash or height.".to_owned(),
|
||||||
|
))?;
|
||||||
}
|
}
|
||||||
let vec = util::from_hex(input).unwrap();
|
let vec = util::from_hex(input).unwrap();
|
||||||
Ok(Hash::from_vec(vec))
|
Ok(Hash::from_vec(vec))
|
||||||
|
@ -545,7 +549,8 @@ impl Handler for BlockHandler {
|
||||||
path_elems.pop();
|
path_elems.pop();
|
||||||
}
|
}
|
||||||
let el = *path_elems.last().unwrap();
|
let el = *path_elems.last().unwrap();
|
||||||
let h = try!(self.parse_input(el.to_string()));
|
let h = self.parse_input(el.to_string())
|
||||||
|
.map_err(|e| IronError::new(Fail::compat(e), status::BadRequest))?;
|
||||||
|
|
||||||
let mut compact = false;
|
let mut compact = false;
|
||||||
if let Ok(params) = req.get_ref::<UrlEncodedQuery>() {
|
if let Ok(params) = req.get_ref::<UrlEncodedQuery>() {
|
||||||
|
@ -555,10 +560,12 @@ impl Handler for BlockHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
if compact {
|
if compact {
|
||||||
let b = try!(self.get_compact_block(&h));
|
let b = self.get_compact_block(&h)
|
||||||
|
.map_err(|e| IronError::new(Fail::compat(e), status::InternalServerError))?;
|
||||||
json_response(&b)
|
json_response(&b)
|
||||||
} else {
|
} else {
|
||||||
let b = try!(self.get_block(&h));
|
let b = self.get_block(&h)
|
||||||
|
.map_err(|e| IronError::new(Fail::compat(e), status::InternalServerError))?;
|
||||||
json_response(&b)
|
json_response(&b)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -603,14 +610,13 @@ where
|
||||||
{
|
{
|
||||||
fn handle(&self, req: &mut Request) -> IronResult<Response> {
|
fn handle(&self, req: &mut Request) -> IronResult<Response> {
|
||||||
let wrapper: TxWrapper = serde_json::from_reader(req.body.by_ref())
|
let wrapper: TxWrapper = serde_json::from_reader(req.body.by_ref())
|
||||||
.map_err(|e| IronError::new(e, status::BadRequest))?;
|
.map_err(|e| IronError::new(Fail::compat(e), status::BadRequest))?;
|
||||||
|
|
||||||
let tx_bin = util::from_hex(wrapper.tx_hex)
|
let tx_bin = util::from_hex(wrapper.tx_hex)
|
||||||
.map_err(|_| Error::Argument(format!("Invalid hex in transaction wrapper.")))?;
|
.map_err(|e| IronError::new(Fail::compat(e), status::BadRequest))?;
|
||||||
|
|
||||||
let tx: Transaction = ser::deserialize(&mut &tx_bin[..]).map_err(|_| {
|
let tx: Transaction = ser::deserialize(&mut &tx_bin[..])
|
||||||
Error::Argument("Could not deserialize transaction, invalid format.".to_string())
|
.map_err(|e| IronError::new(Fail::compat(e), status::BadRequest))?;
|
||||||
})?;
|
|
||||||
|
|
||||||
let source = pool::TxSource {
|
let source = pool::TxSource {
|
||||||
debug_name: "push-api".to_string(),
|
debug_name: "push-api".to_string(),
|
||||||
|
@ -650,7 +656,7 @@ where
|
||||||
Ok(()) => Ok(Response::with(status::Ok)),
|
Ok(()) => Ok(Response::with(status::Ok)),
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
debug!(LOGGER, "error - {:?}", e);
|
debug!(LOGGER, "error - {:?}", e);
|
||||||
Err(IronError::from(Error::Argument(format!("{:?}", e))))
|
Err(IronError::new(Fail::compat(e), status::BadRequest))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,10 @@ extern crate grin_pool as pool;
|
||||||
extern crate grin_store as store;
|
extern crate grin_store as store;
|
||||||
extern crate grin_util as util;
|
extern crate grin_util as util;
|
||||||
|
|
||||||
|
|
||||||
|
extern crate failure;
|
||||||
|
#[macro_use]
|
||||||
|
extern crate failure_derive;
|
||||||
extern crate hyper;
|
extern crate hyper;
|
||||||
extern crate iron;
|
extern crate iron;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
|
|
|
@ -18,64 +18,70 @@
|
||||||
//! To use it, just have your service(s) implement the ApiEndpoint trait and
|
//! To use it, just have your service(s) implement the ApiEndpoint trait and
|
||||||
//! register them on a ApiServer.
|
//! register them on a ApiServer.
|
||||||
|
|
||||||
use std::error;
|
|
||||||
use std::fmt::{self, Display, Formatter};
|
use std::fmt::{self, Display, Formatter};
|
||||||
|
use std::mem;
|
||||||
use std::net::ToSocketAddrs;
|
use std::net::ToSocketAddrs;
|
||||||
use std::string::ToString;
|
use std::string::ToString;
|
||||||
use std::mem;
|
|
||||||
|
|
||||||
|
use failure::{Backtrace, Context, Fail, ResultExt};
|
||||||
|
use iron::middleware::Handler;
|
||||||
use iron::prelude::*;
|
use iron::prelude::*;
|
||||||
use iron::{status, Listening};
|
use iron::{status, Listening};
|
||||||
use iron::middleware::Handler;
|
|
||||||
use router::Router;
|
|
||||||
use mount::Mount;
|
use mount::Mount;
|
||||||
|
use router::Router;
|
||||||
|
|
||||||
use store;
|
use store;
|
||||||
|
|
||||||
/// Errors that can be returned by an ApiEndpoint implementation.
|
/// Errors that can be returned by an ApiEndpoint implementation.
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum Error {
|
pub struct Error {
|
||||||
|
inner: Context<ErrorKind>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Eq, PartialEq, Debug, Fail)]
|
||||||
|
pub enum ErrorKind {
|
||||||
|
#[fail(display = "Internal error: {}", _0)]
|
||||||
Internal(String),
|
Internal(String),
|
||||||
|
#[fail(display = "Bad arguments: {}", _0)]
|
||||||
Argument(String),
|
Argument(String),
|
||||||
|
#[fail(display = "Not found.")]
|
||||||
NotFound,
|
NotFound,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Fail for Error {
|
||||||
|
fn cause(&self) -> Option<&Fail> {
|
||||||
|
self.inner.cause()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn backtrace(&self) -> Option<&Backtrace> {
|
||||||
|
self.inner.backtrace()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Display for Error {
|
impl Display for Error {
|
||||||
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
match *self {
|
Display::fmt(&self.inner, f)
|
||||||
Error::Argument(ref s) => write!(f, "Bad arguments: {}", s),
|
}
|
||||||
Error::Internal(ref s) => write!(f, "Internal error: {}", s),
|
}
|
||||||
Error::NotFound => write!(f, "Not found."),
|
|
||||||
|
impl Error {
|
||||||
|
pub fn kind(&self) -> &ErrorKind {
|
||||||
|
self.inner.get_context()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<ErrorKind> for Error {
|
||||||
|
fn from(kind: ErrorKind) -> Error {
|
||||||
|
Error {
|
||||||
|
inner: Context::new(kind),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl error::Error for Error {
|
impl From<Context<ErrorKind>> for Error {
|
||||||
fn description(&self) -> &str {
|
fn from(inner: Context<ErrorKind>) -> Error {
|
||||||
match *self {
|
Error { inner: inner }
|
||||||
Error::Argument(_) => "Bad arguments.",
|
|
||||||
Error::Internal(_) => "Internal error.",
|
|
||||||
Error::NotFound => "Not found.",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Error> for IronError {
|
|
||||||
fn from(e: Error) -> IronError {
|
|
||||||
match e {
|
|
||||||
Error::Argument(_) => IronError::new(e, status::Status::BadRequest),
|
|
||||||
Error::Internal(_) => IronError::new(e, status::Status::InternalServerError),
|
|
||||||
Error::NotFound => IronError::new(e, status::Status::NotFound),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<store::Error> for Error {
|
|
||||||
fn from(e: store::Error) -> Error {
|
|
||||||
match e {
|
|
||||||
store::Error::NotFoundErr => Error::NotFound,
|
|
||||||
_ => Error::Internal(e.to_string()),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,19 +14,19 @@
|
||||||
|
|
||||||
//! Base types that the block chain pipeline requires.
|
//! Base types that the block chain pipeline requires.
|
||||||
|
|
||||||
use std::io;
|
use std::{error, fmt, io};
|
||||||
|
|
||||||
use util::secp;
|
use util::secp;
|
||||||
use util::secp::pedersen::Commitment;
|
use util::secp::pedersen::Commitment;
|
||||||
|
|
||||||
use grin_store as store;
|
|
||||||
use core::core::{block, transaction, Block, BlockHeader};
|
|
||||||
use core::core::hash::{Hash, Hashed};
|
use core::core::hash::{Hash, Hashed};
|
||||||
use core::core::target::Difficulty;
|
use core::core::target::Difficulty;
|
||||||
|
use core::core::{block, transaction, Block, BlockHeader};
|
||||||
use core::ser::{self, Readable, Reader, Writeable, Writer};
|
use core::ser::{self, Readable, Reader, Writeable, Writer};
|
||||||
use keychain;
|
use grin_store as store;
|
||||||
use grin_store;
|
use grin_store;
|
||||||
use grin_store::pmmr::PMMRFileMetadata;
|
use grin_store::pmmr::PMMRFileMetadata;
|
||||||
|
use keychain;
|
||||||
|
|
||||||
bitflags! {
|
bitflags! {
|
||||||
/// Options for block validation
|
/// Options for block validation
|
||||||
|
@ -109,6 +109,22 @@ pub enum Error {
|
||||||
Other(String),
|
Other(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl error::Error for Error {
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
match *self {
|
||||||
|
_ => "some kind of chain error",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for Error {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
match *self {
|
||||||
|
_ => write!(f, "some kind of chain error"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl From<grin_store::Error> for Error {
|
impl From<grin_store::Error> for Error {
|
||||||
fn from(e: grin_store::Error) -> Error {
|
fn from(e: grin_store::Error) -> Error {
|
||||||
Error::StoreErr(e, "wrapped".to_owned())
|
Error::StoreErr(e, "wrapped".to_owned())
|
||||||
|
@ -285,7 +301,8 @@ pub trait ChainStore: Send + Sync {
|
||||||
fn delete_header_by_height(&self, height: u64) -> Result<(), store::Error>;
|
fn delete_header_by_height(&self, height: u64) -> Result<(), store::Error>;
|
||||||
|
|
||||||
/// Is the block header on the current chain?
|
/// Is the block header on the current chain?
|
||||||
/// Use the header_by_height index to verify the block header is where we think it is.
|
/// Use the header_by_height index to verify the block header is where we
|
||||||
|
/// think it is.
|
||||||
fn is_on_current_chain(&self, header: &BlockHeader) -> Result<(), store::Error>;
|
fn is_on_current_chain(&self, header: &BlockHeader) -> Result<(), store::Error>;
|
||||||
|
|
||||||
/// Saves the position of an output, represented by its commitment, in the
|
/// Saves the position of an output, represented by its commitment, in the
|
||||||
|
@ -299,8 +316,8 @@ pub trait ChainStore: Send + Sync {
|
||||||
/// Deletes the MMR position of an output.
|
/// Deletes the MMR position of an output.
|
||||||
fn delete_output_pos(&self, commit: &[u8]) -> Result<(), store::Error>;
|
fn delete_output_pos(&self, commit: &[u8]) -> Result<(), store::Error>;
|
||||||
|
|
||||||
/// Saves a marker associated with a block recording the MMR positions of its
|
/// Saves a marker associated with a block recording the MMR positions of
|
||||||
/// last elements.
|
/// its last elements.
|
||||||
fn save_block_marker(&self, bh: &Hash, marker: &(u64, u64)) -> Result<(), store::Error>;
|
fn save_block_marker(&self, bh: &Hash, marker: &(u64, u64)) -> Result<(), store::Error>;
|
||||||
|
|
||||||
/// Retrieves a block marker from a block hash.
|
/// Retrieves a block marker from a block hash.
|
||||||
|
|
|
@ -15,18 +15,18 @@
|
||||||
//! The primary module containing the implementations of the transaction pool
|
//! The primary module containing the implementations of the transaction pool
|
||||||
//! and its top-level members.
|
//! and its top-level members.
|
||||||
|
|
||||||
use std::vec::Vec;
|
|
||||||
use std::collections::{HashMap, HashSet};
|
use std::collections::{HashMap, HashSet};
|
||||||
use std::iter::Iterator;
|
use std::iter::Iterator;
|
||||||
use std::fmt;
|
use std::vec::Vec;
|
||||||
|
use std::{error, fmt};
|
||||||
|
|
||||||
use util::secp::pedersen::Commitment;
|
use util::secp::pedersen::Commitment;
|
||||||
|
|
||||||
pub use graph;
|
pub use graph;
|
||||||
|
|
||||||
use core::consensus;
|
use core::consensus;
|
||||||
use core::core::{block, hash, transaction};
|
|
||||||
use core::core::transaction::{Input, OutputIdentifier};
|
use core::core::transaction::{Input, OutputIdentifier};
|
||||||
|
use core::core::{block, hash, transaction};
|
||||||
|
|
||||||
/// Transaction pool configuration
|
/// Transaction pool configuration
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
|
@ -162,14 +162,30 @@ pub enum PoolError {
|
||||||
LowFeeTransaction(u64),
|
LowFeeTransaction(u64),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl error::Error for PoolError {
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
match *self {
|
||||||
|
_ => "some kind of pool error",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for PoolError {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
match *self {
|
||||||
|
_ => write!(f, "some kind of pool error"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Interface that the pool requires from a blockchain implementation.
|
/// Interface that the pool requires from a blockchain implementation.
|
||||||
pub trait BlockChain {
|
pub trait BlockChain {
|
||||||
/// Get an unspent output by its commitment. Will return an error if the output
|
/// Get an unspent output by its commitment. Will return an error if the
|
||||||
/// is spent or if it doesn't exist. The blockchain is expected to produce
|
/// output is spent or if it doesn't exist. The blockchain is expected to
|
||||||
/// a result with its current view of the most worked chain, ignoring
|
/// produce a result with its current view of the most worked chain,
|
||||||
/// orphans, etc.
|
/// ignoring orphans, etc.
|
||||||
/// We do not maintain outputs themselves. The only information we have is the
|
/// We do not maintain outputs themselves. The only information we have is
|
||||||
/// hash from the output MMR.
|
/// the hash from the output MMR.
|
||||||
fn is_unspent(&self, output_ref: &OutputIdentifier) -> Result<hash::Hash, PoolError>;
|
fn is_unspent(&self, output_ref: &OutputIdentifier) -> Result<hash::Hash, PoolError>;
|
||||||
|
|
||||||
/// Check if an output being spent by the input has sufficiently matured.
|
/// Check if an output being spent by the input has sufficiently matured.
|
||||||
|
@ -188,8 +204,8 @@ pub trait PoolAdapter: Send + Sync {
|
||||||
/// The transaction pool has accepted this transactions as valid and added
|
/// The transaction pool has accepted this transactions as valid and added
|
||||||
/// it to its internal cache.
|
/// it to its internal cache.
|
||||||
fn tx_accepted(&self, tx: &transaction::Transaction);
|
fn tx_accepted(&self, tx: &transaction::Transaction);
|
||||||
/// The stem transaction pool has accepted this transactions as valid and added
|
/// The stem transaction pool has accepted this transactions as valid and
|
||||||
/// it to its internal cache.
|
/// added it to its internal cache.
|
||||||
fn stem_tx_accepted(&self, tx: &transaction::Transaction);
|
fn stem_tx_accepted(&self, tx: &transaction::Transaction);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,19 +12,19 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
use iron::prelude::*;
|
use bodyparser;
|
||||||
use iron::Handler;
|
use iron::Handler;
|
||||||
|
use iron::prelude::*;
|
||||||
use iron::status;
|
use iron::status;
|
||||||
use serde_json;
|
use serde_json;
|
||||||
use bodyparser;
|
|
||||||
|
|
||||||
use receiver::receive_coinbase;
|
|
||||||
use core::ser;
|
|
||||||
use api;
|
use api;
|
||||||
|
use core::ser;
|
||||||
|
use failure::{Fail, ResultExt};
|
||||||
use keychain::Keychain;
|
use keychain::Keychain;
|
||||||
|
use receiver::receive_coinbase;
|
||||||
use types::*;
|
use types::*;
|
||||||
use util;
|
use util;
|
||||||
use failure::{Fail, ResultExt};
|
|
||||||
|
|
||||||
pub struct CoinbaseHandler {
|
pub struct CoinbaseHandler {
|
||||||
pub config: WalletConfig,
|
pub config: WalletConfig,
|
||||||
|
@ -33,22 +33,15 @@ pub struct CoinbaseHandler {
|
||||||
|
|
||||||
impl CoinbaseHandler {
|
impl CoinbaseHandler {
|
||||||
fn build_coinbase(&self, block_fees: &BlockFees) -> Result<CbData, Error> {
|
fn build_coinbase(&self, block_fees: &BlockFees) -> Result<CbData, Error> {
|
||||||
let (out, kern, block_fees) = receive_coinbase(&self.config, &self.keychain, block_fees)
|
let (out, kern, block_fees) =
|
||||||
.map_err(|e| api::Error::Internal(format!("Error building coinbase: {:?}", e)))
|
receive_coinbase(&self.config, &self.keychain, block_fees).context(ErrorKind::Node)?;
|
||||||
.context(ErrorKind::Node)?;
|
|
||||||
|
|
||||||
let out_bin = ser::ser_vec(&out)
|
let out_bin = ser::ser_vec(&out).context(ErrorKind::Node)?;
|
||||||
.map_err(|e| api::Error::Internal(format!("Error serializing output: {:?}", e)))
|
|
||||||
.context(ErrorKind::Node)?;
|
|
||||||
|
|
||||||
let kern_bin = ser::ser_vec(&kern)
|
let kern_bin = ser::ser_vec(&kern).context(ErrorKind::Node)?;
|
||||||
.map_err(|e| api::Error::Internal(format!("Error serializing kernel: {:?}", e)))
|
|
||||||
.context(ErrorKind::Node)?;
|
|
||||||
|
|
||||||
let key_id_bin = match block_fees.key_id {
|
let key_id_bin = match block_fees.key_id {
|
||||||
Some(key_id) => ser::ser_vec(&key_id)
|
Some(key_id) => ser::ser_vec(&key_id).context(ErrorKind::Node)?,
|
||||||
.map_err(|e| api::Error::Internal(format!("Error serializing kernel: {:?}", e)))
|
|
||||||
.context(ErrorKind::Node)?,
|
|
||||||
None => vec![],
|
None => vec![],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -17,8 +17,8 @@
|
||||||
//! wallet server that's running at all time is required in many cases.
|
//! wallet server that's running at all time is required in many cases.
|
||||||
|
|
||||||
use bodyparser;
|
use bodyparser;
|
||||||
use iron::prelude::*;
|
|
||||||
use iron::Handler;
|
use iron::Handler;
|
||||||
|
use iron::prelude::*;
|
||||||
use iron::status;
|
use iron::status;
|
||||||
use serde_json;
|
use serde_json;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
@ -27,11 +27,11 @@ use api;
|
||||||
use core::consensus::reward;
|
use core::consensus::reward;
|
||||||
use core::core::{amount_to_hr_string, build, Block, Committed, Output, Transaction, TxKernel};
|
use core::core::{amount_to_hr_string, build, Block, Committed, Output, Transaction, TxKernel};
|
||||||
use core::{global, ser};
|
use core::{global, ser};
|
||||||
|
use failure::{Fail, ResultExt};
|
||||||
use keychain::{BlindingFactor, Identifier, Keychain};
|
use keychain::{BlindingFactor, Identifier, Keychain};
|
||||||
use types::*;
|
use types::*;
|
||||||
use util::{secp, to_hex, LOGGER};
|
|
||||||
use urlencoded::UrlEncodedQuery;
|
use urlencoded::UrlEncodedQuery;
|
||||||
use failure::ResultExt;
|
use util::{secp, to_hex, LOGGER};
|
||||||
|
|
||||||
/// Dummy wrapper for the hex-encoded serialized transaction.
|
/// Dummy wrapper for the hex-encoded serialized transaction.
|
||||||
#[derive(Serialize, Deserialize)]
|
#[derive(Serialize, Deserialize)]
|
||||||
|
@ -42,7 +42,8 @@ pub struct TxWrapper {
|
||||||
/// Receive Part 1 of interactive transactions from sender, Sender Initiation
|
/// Receive Part 1 of interactive transactions from sender, Sender Initiation
|
||||||
/// Return result of part 2, Recipient Initation, to sender
|
/// Return result of part 2, Recipient Initation, to sender
|
||||||
/// -Receiver receives inputs, outputs xS * G and kS * G
|
/// -Receiver receives inputs, outputs xS * G and kS * G
|
||||||
/// -Receiver picks random blinding factors for all outputs being received, computes total blinding
|
/// -Receiver picks random blinding factors for all outputs being received,
|
||||||
|
/// computes total blinding
|
||||||
/// excess xR
|
/// excess xR
|
||||||
/// -Receiver picks random nonce kR
|
/// -Receiver picks random nonce kR
|
||||||
/// -Receiver computes Schnorr challenge e = H(M | kR * G + kS * G)
|
/// -Receiver computes Schnorr challenge e = H(M | kR * G + kS * G)
|
||||||
|
@ -146,7 +147,8 @@ fn handle_sender_initiation(
|
||||||
/// Receive Part 3 of interactive transactions from sender, Sender Confirmation
|
/// Receive Part 3 of interactive transactions from sender, Sender Confirmation
|
||||||
/// Return Ok/Error
|
/// Return Ok/Error
|
||||||
/// -Receiver receives sS
|
/// -Receiver receives sS
|
||||||
/// -Receiver verifies sender's sig, by verifying that kS * G + e *xS * G = sS * G
|
/// -Receiver verifies sender's sig, by verifying that
|
||||||
|
/// kS * G + e *xS * G = sS* G
|
||||||
/// -Receiver calculates final sig as s=(sS+sR, kS * G+kR * G)
|
/// -Receiver calculates final sig as s=(sS+sR, kS * G+kR * G)
|
||||||
/// -Receiver puts into TX kernel:
|
/// -Receiver puts into TX kernel:
|
||||||
///
|
///
|
||||||
|
@ -281,9 +283,8 @@ impl Handler for WalletReceiver {
|
||||||
&partial_tx,
|
&partial_tx,
|
||||||
).map_err(|e| {
|
).map_err(|e| {
|
||||||
error!(LOGGER, "Phase 1 Sender Initiation -> Problematic partial tx, looks like this: {:?}", partial_tx);
|
error!(LOGGER, "Phase 1 Sender Initiation -> Problematic partial tx, looks like this: {:?}", partial_tx);
|
||||||
api::Error::Internal(format!(
|
e.context(api::ErrorKind::Internal(
|
||||||
"Error processing partial transaction: {:?}",
|
"Error processing partial transaction".to_owned(),
|
||||||
e
|
|
||||||
))
|
))
|
||||||
})
|
})
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -298,9 +299,8 @@ impl Handler for WalletReceiver {
|
||||||
fluff,
|
fluff,
|
||||||
).map_err(|e| {
|
).map_err(|e| {
|
||||||
error!(LOGGER, "Phase 3 Sender Confirmation -> Problematic partial tx, looks like this: {:?}", partial_tx);
|
error!(LOGGER, "Phase 3 Sender Confirmation -> Problematic partial tx, looks like this: {:?}", partial_tx);
|
||||||
api::Error::Internal(format!(
|
e.context(api::ErrorKind::Internal(
|
||||||
"Error processing partial transaction: {:?}",
|
"Error processing partial transaction".to_owned(),
|
||||||
e
|
|
||||||
))
|
))
|
||||||
})
|
})
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
Loading…
Reference in a new issue