// Copyright 2015-2020 Parity Technologies (UK) Ltd. // This file is part of OpenEthereum. // OpenEthereum is free software: you can redistribute it and/or modify // 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. // OpenEthereum is distributed in the hope that it will be useful, // 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 // along with OpenEthereum. If not, see . //! Sends HTTP notifications to a list of URLs every time new work is available. extern crate ethash; extern crate fetch; extern crate hyper; extern crate parity_runtime; extern crate url; use self::{ ethash::SeedHashCompute, fetch::{Client as FetchClient, Fetch, Method, Request}, hyper::header::{self, HeaderValue}, parity_runtime::Executor, url::Url, }; use ethereum_types::{H256, U256}; use parking_lot::Mutex; use futures::Future; /// 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); } /// POSTs info about new work to given urls. pub struct WorkPoster { urls: Vec, client: FetchClient, executor: Executor, seed_compute: Mutex, } impl WorkPoster { /// Create new `WorkPoster`. pub fn new(urls: &[String], fetch: FetchClient, executor: Executor) -> Self { let urls = urls .into_iter() .filter_map(|u| match Url::parse(u) { Ok(url) => Some(url), Err(e) => { warn!("Error parsing URL {} : {}", u, e); None } }) .collect(); WorkPoster { client: fetch, executor: executor, urls: urls, seed_compute: Mutex::new(SeedHashCompute::default()), } } } impl NotifyWork for WorkPoster { fn notify(&self, pow_hash: H256, difficulty: U256, number: u64) { // TODO: move this to engine let target = ethash::difficulty_to_boundary(&difficulty); let seed_hash = &self.seed_compute.lock().hash_block_number(number); let seed_hash = H256::from_slice(&seed_hash[..]); let body = format!( r#"{{ "result": ["0x{:x}","0x{:x}","0x{:x}","0x{:x}"] }}"#, pow_hash, seed_hash, target, number ); for u in &self.urls { let u = u.clone(); self.executor.spawn( self.client .fetch( Request::new(u.clone(), Method::POST) .with_header( header::CONTENT_TYPE, HeaderValue::from_static("application/json"), ) .with_body(body.clone()), Default::default(), ) .map_err(move |e| { warn!("Error sending HTTP notification to {} : {}, retrying", u, e); }) .map(|_| ()), ); } } }