mirror of
https://github.com/mimblewimble/grin.git
synced 2025-02-08 12:21:09 +03:00
* Initial Dandelion Commit * Changed stem_tx_pool to tx_stempool * Introduction of stem memory pool and stem pool config * Pool push now send to stem memory pool * Add stem transaction functions * Add stem transaction pool * Drastically simplified code structure * Add monitor transactions * Add Dandelion monitor and remove transactions from stempool * Add peer relay monitor * Reconcile block with stempool * Fix total size bug * Add fluff option for pool push * Added details on dandelion monitor * Fix issue with missing parent * Child transaction with stempool parent are now forced stem * Update Dandelion Relay from outgoing peers * Fix missing pool reconciliation * Add the ability to fluff a transaction directly * Fix tests for Dandelion * Missing send_stem_transaction method... * Add fluff handler for wallet * Add logger when successfully updated Dandelion relay * Launch transaction monitor last * Fix dandelion relay misplaced * Add logging and updating for stempool * Additionnal check for stem transaction * Fix 2 Locks in a row
129 lines
3.9 KiB
Rust
129 lines
3.9 KiB
Rust
// Copyright 2018 The Grin Developers
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use std::time;
|
|
use std::ops::FnMut;
|
|
|
|
use futures::{Future, Stream};
|
|
use failure::ResultExt;
|
|
use hyper;
|
|
use hyper::{Method, Request};
|
|
use hyper::header::ContentType;
|
|
use tokio_core::reactor;
|
|
use tokio_retry::Retry;
|
|
use tokio_retry::strategy::FibonacciBackoff;
|
|
use serde_json;
|
|
|
|
use types::*;
|
|
use util::LOGGER;
|
|
use std::io;
|
|
|
|
/// Call the wallet API to create a coinbase output for the given block_fees.
|
|
/// Will retry based on default "retry forever with backoff" behavior.
|
|
pub fn create_coinbase(url: &str, block_fees: &BlockFees) -> Result<CbData, Error> {
|
|
let mut has_error = false;
|
|
|
|
retry_backoff_forever(|| {
|
|
let res = single_create_coinbase(&url, &block_fees);
|
|
if let Err(_) = res {
|
|
has_error = true;
|
|
error!(
|
|
LOGGER,
|
|
"Failed to get coinbase from {}. Run grin wallet listen", url
|
|
);
|
|
}
|
|
if has_error {
|
|
error!(LOGGER, "Successfully received coinbase from {}", url);
|
|
}
|
|
res
|
|
})
|
|
}
|
|
|
|
/// Runs the specified function wrapped in some basic retry logic.
|
|
fn retry_backoff_forever<F, R>(f: F) -> Result<R, Error>
|
|
where
|
|
F: FnMut() -> Result<R, Error>,
|
|
{
|
|
let mut core =
|
|
reactor::Core::new().context(ErrorKind::GenericError("Could not create reactor"))?;
|
|
let retry_strategy =
|
|
FibonacciBackoff::from_millis(100).max_delay(time::Duration::from_secs(10));
|
|
let retry_future = Retry::spawn(core.handle(), retry_strategy, f);
|
|
let res = core.run(retry_future).unwrap();
|
|
Ok(res)
|
|
}
|
|
|
|
pub fn send_partial_tx(url: &str, partial_tx: &PartialTx, fluff: bool) -> Result<PartialTx, Error> {
|
|
single_send_partial_tx(url, partial_tx, fluff)
|
|
}
|
|
|
|
fn single_send_partial_tx(
|
|
url: &str,
|
|
partial_tx: &PartialTx,
|
|
fluff: bool,
|
|
) -> Result<PartialTx, Error> {
|
|
let mut core = reactor::Core::new().context(ErrorKind::Hyper)?;
|
|
let client = hyper::Client::new(&core.handle());
|
|
|
|
// In case we want to do an express send
|
|
let mut url_pool = url.to_owned();
|
|
if fluff {
|
|
url_pool = format!("{}{}", url, "?fluff");
|
|
}
|
|
|
|
let mut req = Request::new(
|
|
Method::Post,
|
|
url_pool.parse::<hyper::Uri>().context(ErrorKind::Hyper)?,
|
|
);
|
|
req.headers_mut().set(ContentType::json());
|
|
let json = serde_json::to_string(&partial_tx).context(ErrorKind::Hyper)?;
|
|
req.set_body(json);
|
|
|
|
let work = client.request(req).and_then(|res| {
|
|
res.body().concat2().and_then(move |body| {
|
|
let partial_tx: PartialTx =
|
|
serde_json::from_slice(&body).map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
|
|
Ok(partial_tx)
|
|
})
|
|
});
|
|
let res = core.run(work).context(ErrorKind::Hyper)?;
|
|
Ok(res)
|
|
}
|
|
|
|
/// Makes a single request to the wallet API to create a new coinbase output.
|
|
fn single_create_coinbase(url: &str, block_fees: &BlockFees) -> Result<CbData, Error> {
|
|
let mut core =
|
|
reactor::Core::new().context(ErrorKind::GenericError("Could not create reactor"))?;
|
|
let client = hyper::Client::new(&core.handle());
|
|
|
|
let mut req = Request::new(
|
|
Method::Post,
|
|
url.parse::<hyper::Uri>().context(ErrorKind::Uri)?,
|
|
);
|
|
req.headers_mut().set(ContentType::json());
|
|
let json = serde_json::to_string(&block_fees).context(ErrorKind::Format)?;
|
|
req.set_body(json);
|
|
|
|
let work = client.request(req).and_then(|res| {
|
|
res.body().concat2().and_then(move |body| {
|
|
let coinbase: CbData =
|
|
serde_json::from_slice(&body).map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
|
|
Ok(coinbase)
|
|
})
|
|
});
|
|
|
|
let res = core.run(work)
|
|
.context(ErrorKind::GenericError("Could not run core"))?;
|
|
Ok(res)
|
|
}
|