use std::collections::{HashMap, VecDeque}; use color_eyre::Result; use crossterm::event::{KeyCode, KeyEvent}; use primitive_types::H256; use ratatui::{ layout::{Alignment, Rect}, prelude::*, style::Style, text::Line, widgets::{Block, BorderType, Paragraph}, Frame }; use sp_core::crypto::{AccountId32, Ss58Codec, Ss58AddressFormat}; use codec::Decode; use super::Component; use crate::{ types::CasperExtrinsicDetails, CasperAccountId, action::Action, app::Mode, palette::StylePalette, }; struct BlockInfo { block_number: u32, finalized: bool, } #[derive(Default)] pub struct ExplorerBlocks { blocks: VecDeque, block_headers: HashMap, authors: HashMap, extrinsics: HashMap>, palette: StylePalette, current_block_digit_length: u32, is_active: bool, used_paragraph_index: usize, used_block_number: Option, used_ext_index: Option<(H256, usize)>, } impl ExplorerBlocks { const MAX_BLOCKS: usize = 50; const LENGTH_OF_BLOCK_HASH: u16 = 13; const LENGTH_OF_ADDRESS: u16 = 49; const TOTAL_OFFSETS: u16 = 18; fn update_block_author( &mut self, hash: H256, maybe_author: Option, ) -> Result<()> { if let Some(author) = maybe_author { self.authors.insert(hash, author); } Ok(()) } fn update_latest_block_info( &mut self, hash: H256, block_number: u32, extrinsics: Vec, ) -> Result<()> { let front_block_number = match self.blocks.front() { Some(block_info) => block_info.block_number, None => 0, }; if front_block_number < block_number { self.blocks.push_front(BlockInfo { block_number, finalized: false, }); self.extrinsics.insert(hash, extrinsics); self.block_headers.insert(block_number, hash); let block_length = block_number.checked_ilog10().unwrap_or(0) + 1; if self.current_block_digit_length < block_length { self.current_block_digit_length = block_length; } if self.blocks.len() > Self::MAX_BLOCKS { if let Some(block) = self.blocks.pop_back() { if let Some(hash) = self.block_headers.remove(&block.block_number) { self.extrinsics.remove(&hash); self.authors.remove(&hash); } } } } Ok(()) } fn update_finalized_block_info( &mut self, _hash: H256, block_number: u32, _extrinsics: Vec, ) -> Result<()> { for idx in 0..self.blocks.len() { if self.blocks[idx].finalized { break; } else if self.blocks[idx].block_number > block_number { continue; } else { self.blocks[idx].finalized = true; } } Ok(()) } fn prepare_block_line_info(&self, current_block: &BlockInfo, width: u16) -> Line { let block_number_length = self .current_block_digit_length .max(current_block.block_number.checked_ilog10().unwrap_or(0) + 1) as usize; let free_space = width .saturating_sub(block_number_length as u16) .saturating_sub(Self::TOTAL_OFFSETS); let default_hash = H256::repeat_byte(69u8); let hash = self .block_headers .get(¤t_block.block_number) .unwrap_or(&default_hash); let author = self .authors .get(&hash) .map_or(String::from("..."), |author| { let extended_author = AccountId32::decode(&mut author.as_ref()) .expect("author should be valid AccountId32; qed"); extended_author.to_ss58check_with_version(Ss58AddressFormat::custom(1996)) }); if free_space < Self::LENGTH_OF_BLOCK_HASH + Self::LENGTH_OF_ADDRESS { let len_for_author = free_space * Self::LENGTH_OF_ADDRESS / (Self::LENGTH_OF_BLOCK_HASH + Self::LENGTH_OF_ADDRESS); if &author == "..." { Line::raw(format!("{:^left$}| {} | {:^right$}", current_block.block_number, hash.to_string(), author, left=block_number_length, right=(len_for_author + 2) as usize)) } else { Line::raw(format!("{} | {} | {}", current_block.block_number, hash.to_string(), format!("{}...", &author[..(len_for_author) as usize]))) } } else { let total_space_used = block_number_length as u16 + Self::LENGTH_OF_BLOCK_HASH + Self::LENGTH_OF_ADDRESS; let str_length = width.saturating_sub(2).saturating_sub(total_space_used) as usize / 3; Line::raw(format!("{:^length_block_number$}|{:^length_hash$}|{:^length_author$}", current_block.block_number, hash.to_string(), author, length_block_number=str_length+block_number_length, length_hash=str_length+(Self::LENGTH_OF_BLOCK_HASH as usize), length_author=str_length+(Self::LENGTH_OF_ADDRESS as usize))) } } fn prepare_block_lines(&mut self, rect: Rect) -> Vec { let width = rect.as_size().width; let total_length = rect.as_size().height as usize - 2; let mut items = Vec::new(); let active_style = self.palette.create_text_style(true); let latest_style = self.palette.create_text_style(false); let finalized_style = Style::new().fg(self.palette.foreground_hover()); let start_index = match self.used_block_number { Some(used_block) if total_length < self.blocks.len() => { self.blocks .iter() .position(|info| info.block_number == used_block) .unwrap_or_default() .saturating_add(1) .saturating_sub(total_length) }, _ => 0, }; for (idx, current_block_info) in self.blocks.iter().skip(start_index).enumerate() { if idx == total_length { break; } let style = match self.used_block_number { Some(used_block) if current_block_info.block_number == used_block => active_style, _ => { if current_block_info.finalized { finalized_style } else { latest_style } } }; items.push(self.prepare_block_line_info(¤t_block_info, width).style(style)); } items } fn prepare_ext_line_info( &self, index: usize, width: u16, pallet_name: &str, variant_name: &str, hash: &str, ) -> Line { let index_length = 4; // always 4, two digits and two spaces let hash_length = Self::LENGTH_OF_BLOCK_HASH as usize + 2; let pallet_name_length = pallet_name.len(); let variant_name_length = variant_name.len(); let offset_variant = (width as usize) .saturating_sub(index_length) .saturating_sub(pallet_name_length) .saturating_sub(variant_name_length) .saturating_sub(hash_length) .saturating_sub(2) / 2; let offset_pallet = if offset_variant % 2 == 0 { offset_variant } else { offset_variant + 1 }; Line::from(format!("{:^index_length$}{:>pallet_name_offset$}::{: Vec { let width = rect.as_size().width; let mut total_length = rect.as_size().height - 2; let mut items = Vec::new(); let normal_style = self.palette.create_text_style(false); let active_style = self.palette.create_text_style(true); if let Some(used_block_number) = self.used_block_number { let default_hash = H256::repeat_byte(69u8); let hash = self.block_headers .get(&used_block_number) .unwrap_or(&default_hash); if let Some(exts) = self.extrinsics.get(&hash) { for (index, ext) in exts.iter().enumerate() { if total_length == 0 { break; } let style = if let Some((_, used_ext_index)) = self.used_ext_index { if index == used_ext_index { active_style } else { normal_style } } else { normal_style }; items.push(self.prepare_ext_line_info( index, width.saturating_sub(2), &ext.pallet_name, &ext.variant_name, &ext.hash.to_string()).style(style)); total_length -= 1; } } } items } fn prepare_event_lines(&mut self, rect: Rect) -> Line { let _width = rect.as_size().width; match self.used_ext_index { Some((header, used_index)) if self.extrinsics.get(&header).is_some() => { let exts = self.extrinsics .get(&header) .expect("extrinsics should exists, checked before"); let details = exts .get(used_index) .map_or(Vec::new(), |ext| ext.field_bytes.clone()); Line::from(format!("{}", hex::encode(&details))) }, _ => Line::from(""), }.style(self.palette.create_text_style(false)) } fn move_right(&mut self) { let new_index = self.used_paragraph_index + 1; if new_index < 2 { self.used_paragraph_index = new_index; } } fn move_left(&mut self) { self.used_paragraph_index = self .used_paragraph_index .saturating_sub(1); self.used_ext_index = None; } fn move_down(&mut self) { if self.used_paragraph_index == 0 { self.move_down_blocks(); } else { self.move_down_extrinsics(); } } fn move_up(&mut self) { if self.used_paragraph_index == 0 { self.move_up_blocks(); } else { self.move_up_extrinsics(); } } fn move_up_extrinsics(&mut self) { match &self.used_ext_index { Some((header, used_index)) => { let new_index = used_index.saturating_sub(1); if let Some(exts) = self.extrinsics.get(header) { if exts.get(new_index).is_some() { self.used_ext_index = Some((*header, new_index)); } } }, None => { self.used_ext_index = self.used_block_number .map(|block_number| { let header = self.block_headers .get(&block_number) .expect("header exists for each block number; qed"); self.extrinsics.get(&header).map(|_| (*header, 0usize)) }) .flatten() } } } fn move_up_blocks(&mut self) { self.used_block_number = match &self.used_block_number { Some(block_number) => { Some(self.blocks .iter() .find(|info| info.block_number == block_number + 1) .map(|info| info.block_number) .unwrap_or(*block_number)) }, None => self.blocks.front().map(|info| info.block_number), } } fn move_down_extrinsics(&mut self) { match &self.used_ext_index { Some((header, used_index)) => { let new_index = used_index + 1; if let Some(exts) = self.extrinsics.get(&header) { if new_index < exts.len() && exts.get(new_index).is_some() { self.used_ext_index = Some((*header, new_index)); } } }, None => { self.used_ext_index = self.used_block_number .map(|block_number| { let header = self.block_headers .get(&block_number) .expect("header exists for each block number; qed"); self.extrinsics.get(&header).map(|_| (*header, 0usize)) }) .flatten() } } } fn move_down_blocks(&mut self) { self.used_block_number = match &self.used_block_number { Some(block_number) => { Some(self.blocks .iter() .find(|info| info.block_number == block_number.saturating_sub(1)) .map(|info| info.block_number) .unwrap_or(*block_number)) }, None => { self.blocks.front().map(|info| info.block_number) } } } fn set_active(&mut self) -> Result<()> { self.is_active = true; Ok(()) } fn unset_active(&mut self) -> Result<()> { self.is_active = false; Ok(()) } fn prepare_blocks_paragraph( &mut self, place: Rect, border_style: Color, border_type: BorderType, ) -> Paragraph { let title_style = self.palette.create_title_style(); Paragraph::new(self.prepare_block_lines(place)) .block(Block::bordered() .border_style(border_style) .border_type(border_type) .title_alignment(Alignment::Right) .title_style(title_style) .title("Blocks")) .alignment(Alignment::Center) } fn prepare_extrinsics_paragraph( &mut self, place: Rect, border_style: Color, border_type: BorderType, ) -> Paragraph { let title_style = self.palette.create_title_style(); Paragraph::new(self.prepare_ext_lines(place)) .block(Block::bordered() .border_style(border_style) .border_type(border_type) .title_alignment(Alignment::Right) .title_style(title_style) .title("Transactions")) .alignment(Alignment::Center) } fn prepare_event_paragraph( &mut self, place: Rect, border_style: Color, border_type: BorderType, ) -> Paragraph { let title_style = self.palette.create_title_style(); Paragraph::new(self.prepare_event_lines(place)) .block(Block::bordered() .border_style(border_style) .border_type(border_type) .title_alignment(Alignment::Right) .title_style(title_style) .title("Events")) .alignment(Alignment::Center) } } impl Component for ExplorerBlocks { fn handle_key_event(&mut self, key: KeyEvent) -> Result> { match key.code { KeyCode::Char('k') | KeyCode::Up if self.is_active => self.move_up(), KeyCode::Char('j') | KeyCode::Down if self.is_active => self.move_down(), KeyCode::Char('l') | KeyCode::Right if self.is_active => self.move_right(), KeyCode::Char('h') | KeyCode::Left if self.is_active => self.move_left(), KeyCode::Esc => { self.used_block_number = None; self.used_ext_index = None; self.used_paragraph_index = 0; }, _ => {}, }; Ok(None) } fn update(&mut self, action: Action) -> Result> { match action { Action::BestBlockInformation(hash, block_number, extrinsics) => self.update_latest_block_info(hash, block_number, extrinsics)?, Action::FinalizedBlockInformation(hash, block_number, extrinsics) => self.update_finalized_block_info(hash, block_number, extrinsics)?, Action::SetBlockAuthor(hash, maybe_author) => self.update_block_author(hash, maybe_author)?, Action::SetMode(Mode::ExplorerActive) if !self.is_active => self.set_active()?, Action::SetMode(_) if self.is_active => self.unset_active()?, _ => {} }; Ok(None) } fn draw(&mut self, frame: &mut Frame, area: Rect) -> Result<()> { let [blocks_place, ext_place] = super::explorer_scrollbars_layout(area); let [_, _, event_place] = super::explorer_layout(area); let (border_style_block, border_type_block) = self.palette.create_border_style(self.is_active && self.used_paragraph_index == 0); let (border_style_extrinsics, border_type_extrinsics) = self.palette.create_border_style(self.is_active && self.used_paragraph_index == 1); // TODO: never used, revisit let (border_style_event, border_type_event) = self.palette.create_border_style(self.is_active && self.used_paragraph_index == 2); frame.render_widget(self.prepare_blocks_paragraph(blocks_place, border_style_block, border_type_block), blocks_place); frame.render_widget(self.prepare_extrinsics_paragraph(ext_place, border_style_extrinsics, border_type_extrinsics), ext_place); frame.render_widget(self.prepare_event_paragraph(event_place, border_style_event, border_type_event), event_place); Ok(()) } }