// 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}; 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::*; 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/ /// GET /v1/headers/ /// GET /v1/headers/ /// pub struct HeaderHandler { pub chain: Weak, } impl HeaderHandler { fn get_header(&self, input: String) -> Result { // 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 { 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) -> ResponseFuture { let el = match req.uri().path().trim_right_matches('/').rsplit('/').next() { 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/ /// GET /v1/blocks/ /// GET /v1/blocks/ /// /// Optionally return results as "compact blocks" by passing "?compact" query /// param GET /v1/blocks/?compact pub struct BlockHandler { pub chain: Weak, } impl BlockHandler { fn get_block(&self, h: &Hash) -> Result { 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 { 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 { 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(), ))?; } Ok(()) } impl Handler for BlockHandler { fn get(&self, req: Request) -> ResponseFuture { let el = match req.uri().path().trim_right_matches('/').rsplit('/').next() { 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)) } } }