2020-09-22 14:53:52 +02:00
|
|
|
// Copyright 2015-2020 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of OpenEthereum.
|
2016-06-29 20:07:21 +02:00
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is free software: you can redistribute it and/or modify
|
2016-06-29 20:07:21 +02:00
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is distributed in the hope that it will be useful,
|
2016-06-29 20:07:21 +02:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
2020-09-22 14:53:52 +02:00
|
|
|
// along with OpenEthereum. If not, see <http://www.gnu.org/licenses/>.
|
2016-06-29 20:07:21 +02:00
|
|
|
|
2018-01-11 17:49:10 +01:00
|
|
|
//! Sends HTTP notifications to a list of URLs every time new work is available.
|
|
|
|
|
|
|
|
extern crate ethash;
|
2018-04-10 13:51:29 +02:00
|
|
|
extern crate fetch;
|
2016-06-29 20:07:21 +02:00
|
|
|
extern crate hyper;
|
2018-10-22 09:40:50 +02:00
|
|
|
extern crate parity_runtime;
|
2018-04-10 13:51:29 +02:00
|
|
|
extern crate url;
|
2016-06-29 20:07:21 +02:00
|
|
|
|
2018-04-10 13:51:29 +02:00
|
|
|
use self::{
|
2018-01-11 17:49:10 +01:00
|
|
|
ethash::SeedHashCompute,
|
2018-04-10 13:51:29 +02:00
|
|
|
fetch::{Client as FetchClient, Fetch, Method, Request},
|
2018-10-22 09:40:50 +02:00
|
|
|
hyper::header::{self, HeaderValue},
|
|
|
|
parity_runtime::Executor,
|
|
|
|
url::Url,
|
|
|
|
};
|
2018-01-11 17:49:10 +01:00
|
|
|
|
2018-01-10 13:35:18 +01:00
|
|
|
use ethereum_types::{H256, U256};
|
2017-09-02 20:09:13 +02:00
|
|
|
use parking_lot::Mutex;
|
2018-10-22 09:40:50 +02:00
|
|
|
|
2018-04-10 13:51:29 +02:00
|
|
|
use futures::Future;
|
2016-06-29 20:07:21 +02:00
|
|
|
|
2017-01-25 11:03:36 +01:00
|
|
|
/// Trait for notifying about new mining work
|
|
|
|
pub trait NotifyWork: Send + Sync {
|
|
|
|
/// Fired when new mining job available
|
|
|
|
fn notify(&self, pow_hash: H256, difficulty: U256, number: u64);
|
|
|
|
}
|
|
|
|
|
2018-01-11 17:49:10 +01:00
|
|
|
/// POSTs info about new work to given urls.
|
2016-06-29 20:07:21 +02:00
|
|
|
pub struct WorkPoster {
|
|
|
|
urls: Vec<Url>,
|
2018-04-10 13:51:29 +02:00
|
|
|
client: FetchClient,
|
2018-10-22 09:40:50 +02:00
|
|
|
executor: Executor,
|
2016-06-29 20:07:21 +02:00
|
|
|
seed_compute: Mutex<SeedHashCompute>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl WorkPoster {
|
2018-01-11 17:49:10 +01:00
|
|
|
/// Create new `WorkPoster`.
|
2018-10-22 09:40:50 +02:00
|
|
|
pub fn new(urls: &[String], fetch: FetchClient, executor: Executor) -> Self {
|
2016-06-29 20:07:21 +02:00
|
|
|
let urls = urls
|
|
|
|
.into_iter()
|
2016-07-26 20:31:25 +02:00
|
|
|
.filter_map(|u| match Url::parse(u) {
|
2016-06-29 20:07:21 +02:00
|
|
|
Ok(url) => Some(url),
|
|
|
|
Err(e) => {
|
|
|
|
warn!("Error parsing URL {} : {}", u, e);
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
WorkPoster {
|
2018-04-10 13:51:29 +02:00
|
|
|
client: fetch,
|
2018-10-22 09:40:50 +02:00
|
|
|
executor: executor,
|
2016-06-29 20:07:21 +02:00
|
|
|
urls: urls,
|
2018-07-09 16:47:58 +02:00
|
|
|
seed_compute: Mutex::new(SeedHashCompute::default()),
|
2016-06-29 20:07:21 +02:00
|
|
|
}
|
|
|
|
}
|
2017-01-25 11:03:36 +01:00
|
|
|
}
|
2016-06-30 10:07:33 +02:00
|
|
|
|
2017-01-25 11:03:36 +01:00
|
|
|
impl NotifyWork for WorkPoster {
|
|
|
|
fn notify(&self, pow_hash: H256, difficulty: U256, number: u64) {
|
2016-06-29 20:07:21 +02:00
|
|
|
// TODO: move this to engine
|
2018-08-10 15:36:19 +02:00
|
|
|
let target = ethash::difficulty_to_boundary(&difficulty);
|
2017-09-25 19:45:33 +02:00
|
|
|
let seed_hash = &self.seed_compute.lock().hash_block_number(number);
|
2016-06-29 20:07:21 +02:00
|
|
|
let seed_hash = H256::from_slice(&seed_hash[..]);
|
2016-07-01 02:21:22 +02:00
|
|
|
let body = format!(
|
2018-02-09 09:32:06 +01:00
|
|
|
r#"{{ "result": ["0x{:x}","0x{:x}","0x{:x}","0x{:x}"] }}"#,
|
|
|
|
pow_hash, seed_hash, target, number
|
2016-07-01 02:21:22 +02:00
|
|
|
);
|
2018-04-10 13:51:29 +02:00
|
|
|
|
2016-06-29 20:07:21 +02:00
|
|
|
for u in &self.urls {
|
2018-04-10 13:51:29 +02:00
|
|
|
let u = u.clone();
|
2018-10-22 09:40:50 +02:00
|
|
|
self.executor.spawn(
|
|
|
|
self.client
|
|
|
|
.fetch(
|
|
|
|
Request::new(u.clone(), Method::POST)
|
|
|
|
.with_header(
|
|
|
|
header::CONTENT_TYPE,
|
|
|
|
HeaderValue::from_static("application/json"),
|
2018-04-10 13:51:29 +02:00
|
|
|
)
|
|
|
|
.with_body(body.clone()),
|
|
|
|
Default::default(),
|
|
|
|
)
|
|
|
|
.map_err(move |e| {
|
2016-06-30 10:07:33 +02:00
|
|
|
warn!("Error sending HTTP notification to {} : {}, retrying", u, e);
|
2018-04-10 13:51:29 +02:00
|
|
|
})
|
|
|
|
.map(|_| ()),
|
|
|
|
);
|
2016-07-01 02:21:22 +02:00
|
|
|
}
|
2020-08-05 06:08:03 +02:00
|
|
|
}
|
2016-06-29 20:07:21 +02:00
|
|
|
}
|