2018-11-03 09:46:12 +03:00
|
|
|
// 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 super::utils::{get_output, w};
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::chain;
|
|
|
|
use crate::core::core::hash::Hash;
|
|
|
|
use crate::core::core::hash::Hashed;
|
|
|
|
use crate::rest::*;
|
|
|
|
use crate::router::{Handler, ResponseFuture};
|
|
|
|
use crate::types::*;
|
|
|
|
use crate::util;
|
|
|
|
use crate::web::*;
|
2018-11-03 09:46:12 +03:00
|
|
|
use failure::ResultExt;
|
|
|
|
use hyper::{Body, Request, StatusCode};
|
|
|
|
use regex::Regex;
|
|
|
|
use std::sync::Weak;
|
|
|
|
|
|
|
|
/// Gets block headers given either a hash or height or an output commit.
|
|
|
|
/// GET /v1/headers/<hash>
|
|
|
|
/// GET /v1/headers/<height>
|
|
|
|
/// GET /v1/headers/<output commit>
|
|
|
|
///
|
|
|
|
pub struct HeaderHandler {
|
|
|
|
pub chain: Weak<chain::Chain>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HeaderHandler {
|
|
|
|
fn get_header(&self, input: String) -> Result<BlockHeaderPrintable, Error> {
|
|
|
|
// will fail quick if the provided isn't a commitment
|
|
|
|
if let Ok(h) = self.get_header_for_output(input.clone()) {
|
|
|
|
return Ok(h);
|
|
|
|
}
|
|
|
|
if let Ok(height) = input.parse() {
|
|
|
|
match w(&self.chain).get_header_by_height(height) {
|
|
|
|
Ok(header) => return Ok(BlockHeaderPrintable::from_header(&header)),
|
|
|
|
Err(_) => return Err(ErrorKind::NotFound)?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
check_block_param(&input)?;
|
|
|
|
let vec = util::from_hex(input)
|
|
|
|
.map_err(|e| ErrorKind::Argument(format!("invalid input: {}", e)))?;
|
|
|
|
let h = Hash::from_vec(&vec);
|
|
|
|
let header = w(&self.chain)
|
|
|
|
.get_block_header(&h)
|
|
|
|
.context(ErrorKind::NotFound)?;
|
|
|
|
Ok(BlockHeaderPrintable::from_header(&header))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_header_for_output(&self, commit_id: String) -> Result<BlockHeaderPrintable, Error> {
|
|
|
|
let oid = get_output(&self.chain, &commit_id)?.1;
|
|
|
|
match w(&self.chain).get_header_for_output(&oid) {
|
|
|
|
Ok(header) => return Ok(BlockHeaderPrintable::from_header(&header)),
|
|
|
|
Err(_) => return Err(ErrorKind::NotFound)?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Handler for HeaderHandler {
|
|
|
|
fn get(&self, req: Request<Body>) -> ResponseFuture {
|
2019-01-20 21:14:36 +03:00
|
|
|
let el = match req.uri().path().trim_right_matches('/').rsplit('/').next() {
|
2018-11-03 09:46:12 +03:00
|
|
|
None => return response(StatusCode::BAD_REQUEST, "invalid url"),
|
|
|
|
Some(el) => el,
|
|
|
|
};
|
|
|
|
result_to_response(self.get_header(el.to_string()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Gets block details given either a hash or an unspent commit
|
|
|
|
/// GET /v1/blocks/<hash>
|
|
|
|
/// GET /v1/blocks/<height>
|
|
|
|
/// GET /v1/blocks/<commit>
|
|
|
|
///
|
|
|
|
/// Optionally return results as "compact blocks" by passing "?compact" query
|
|
|
|
/// param GET /v1/blocks/<hash>?compact
|
|
|
|
pub struct BlockHandler {
|
|
|
|
pub chain: Weak<chain::Chain>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BlockHandler {
|
|
|
|
fn get_block(&self, h: &Hash) -> Result<BlockPrintable, Error> {
|
|
|
|
let block = w(&self.chain).get_block(h).context(ErrorKind::NotFound)?;
|
|
|
|
Ok(BlockPrintable::from_block(&block, w(&self.chain), false))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_compact_block(&self, h: &Hash) -> Result<CompactBlockPrintable, Error> {
|
|
|
|
let block = w(&self.chain).get_block(h).context(ErrorKind::NotFound)?;
|
|
|
|
Ok(CompactBlockPrintable::from_compact_block(
|
|
|
|
&block.into(),
|
|
|
|
w(&self.chain),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to decode the string as a height or a hash.
|
|
|
|
fn parse_input(&self, input: String) -> Result<Hash, Error> {
|
|
|
|
if let Ok(height) = input.parse() {
|
|
|
|
match w(&self.chain).get_header_by_height(height) {
|
|
|
|
Ok(header) => return Ok(header.hash()),
|
|
|
|
Err(_) => return Err(ErrorKind::NotFound)?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
check_block_param(&input)?;
|
|
|
|
let vec = util::from_hex(input)
|
|
|
|
.map_err(|e| ErrorKind::Argument(format!("invalid input: {}", e)))?;
|
|
|
|
Ok(Hash::from_vec(&vec))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_block_param(input: &String) -> Result<(), Error> {
|
|
|
|
lazy_static! {
|
|
|
|
static ref RE: Regex = Regex::new(r"[0-9a-fA-F]{64}").unwrap();
|
|
|
|
}
|
|
|
|
if !RE.is_match(&input) {
|
|
|
|
return Err(ErrorKind::Argument(
|
|
|
|
"Not a valid hash or height.".to_owned(),
|
|
|
|
))?;
|
|
|
|
}
|
2019-01-20 21:14:36 +03:00
|
|
|
Ok(())
|
2018-11-03 09:46:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Handler for BlockHandler {
|
|
|
|
fn get(&self, req: Request<Body>) -> ResponseFuture {
|
2019-01-20 21:14:36 +03:00
|
|
|
let el = match req.uri().path().trim_right_matches('/').rsplit('/').next() {
|
2018-11-03 09:46:12 +03:00
|
|
|
None => return response(StatusCode::BAD_REQUEST, "invalid url"),
|
|
|
|
Some(el) => el,
|
|
|
|
};
|
|
|
|
|
|
|
|
let h = match self.parse_input(el.to_string()) {
|
|
|
|
Err(e) => {
|
|
|
|
return response(
|
|
|
|
StatusCode::BAD_REQUEST,
|
|
|
|
format!("failed to parse input: {}", e),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
Ok(h) => h,
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Some(param) = req.uri().query() {
|
|
|
|
if param == "compact" {
|
|
|
|
result_to_response(self.get_compact_block(&h))
|
|
|
|
} else {
|
|
|
|
response(
|
|
|
|
StatusCode::BAD_REQUEST,
|
|
|
|
format!("unsupported query parameter: {}", param),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result_to_response(self.get_block(&h))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|