LCOV - code coverage report
Current view: top level - pageserver/ctl/src - layers.rs (source / functions) Coverage Total Hit
Test: 465a86b0c1fda0069b3e0f6c1c126e6b635a1f72.info Lines: 0.0 % 155 0
Test Date: 2024-06-25 15:47:26 Functions: 0.0 % 32 0

            Line data    Source code
       1              : use std::path::{Path, PathBuf};
       2              : 
       3              : use anyhow::Result;
       4              : use camino::{Utf8Path, Utf8PathBuf};
       5              : use clap::Subcommand;
       6              : use pageserver::context::{DownloadBehavior, RequestContext};
       7              : use pageserver::task_mgr::TaskKind;
       8              : use pageserver::tenant::block_io::BlockCursor;
       9              : use pageserver::tenant::disk_btree::DiskBtreeReader;
      10              : use pageserver::tenant::storage_layer::delta_layer::{BlobRef, Summary};
      11              : use pageserver::tenant::storage_layer::{delta_layer, image_layer};
      12              : use pageserver::tenant::storage_layer::{DeltaLayer, ImageLayer};
      13              : use pageserver::tenant::{TENANTS_SEGMENT_NAME, TIMELINES_SEGMENT_NAME};
      14              : use pageserver::{page_cache, virtual_file};
      15              : use pageserver::{
      16              :     repository::{Key, KEY_SIZE},
      17              :     tenant::{
      18              :         block_io::FileBlockReader, disk_btree::VisitDirection,
      19              :         storage_layer::delta_layer::DELTA_KEY_SIZE,
      20              :     },
      21              :     virtual_file::VirtualFile,
      22              : };
      23              : use std::fs;
      24              : use utils::bin_ser::BeSer;
      25              : use utils::id::{TenantId, TimelineId};
      26              : 
      27              : use crate::layer_map_analyzer::parse_filename;
      28              : 
      29            0 : #[derive(Subcommand)]
      30              : pub(crate) enum LayerCmd {
      31              :     /// List all tenants and timelines under the pageserver path
      32              :     ///
      33              :     /// Example: `cargo run --bin pagectl layer list .neon/`
      34            0 :     List { path: PathBuf },
      35              :     /// List all layers of a given tenant and timeline
      36              :     ///
      37              :     /// Example: `cargo run --bin pagectl layer list .neon/`
      38              :     ListLayer {
      39            0 :         path: PathBuf,
      40            0 :         tenant: String,
      41            0 :         timeline: String,
      42              :     },
      43              :     /// Dump all information of a layer file
      44              :     DumpLayer {
      45            0 :         path: PathBuf,
      46            0 :         tenant: String,
      47            0 :         timeline: String,
      48              :         /// The id from list-layer command
      49            0 :         id: usize,
      50              :     },
      51              :     RewriteSummary {
      52            0 :         layer_file_path: Utf8PathBuf,
      53              :         #[clap(long)]
      54              :         new_tenant_id: Option<TenantId>,
      55              :         #[clap(long)]
      56              :         new_timeline_id: Option<TimelineId>,
      57              :     },
      58              : }
      59              : 
      60            0 : async fn read_delta_file(path: impl AsRef<Path>, ctx: &RequestContext) -> Result<()> {
      61            0 :     let path = Utf8Path::from_path(path.as_ref()).expect("non-Unicode path");
      62            0 :     virtual_file::init(10, virtual_file::api::IoEngineKind::StdFs);
      63            0 :     page_cache::init(100);
      64            0 :     let file = VirtualFile::open(path, ctx).await?;
      65            0 :     let file_id = page_cache::next_file_id();
      66            0 :     let block_reader = FileBlockReader::new(&file, file_id);
      67            0 :     let summary_blk = block_reader.read_blk(0, ctx).await?;
      68            0 :     let actual_summary = Summary::des_prefix(summary_blk.as_ref())?;
      69            0 :     let tree_reader = DiskBtreeReader::<_, DELTA_KEY_SIZE>::new(
      70            0 :         actual_summary.index_start_blk,
      71            0 :         actual_summary.index_root_blk,
      72            0 :         &block_reader,
      73            0 :     );
      74            0 :     // TODO(chi): dedup w/ `delta_layer.rs` by exposing the API.
      75            0 :     let mut all = vec![];
      76            0 :     tree_reader
      77            0 :         .visit(
      78            0 :             &[0u8; DELTA_KEY_SIZE],
      79            0 :             VisitDirection::Forwards,
      80            0 :             |key, value_offset| {
      81            0 :                 let curr = Key::from_slice(&key[..KEY_SIZE]);
      82            0 :                 all.push((curr, BlobRef(value_offset)));
      83            0 :                 true
      84            0 :             },
      85            0 :             ctx,
      86            0 :         )
      87            0 :         .await?;
      88            0 :     let cursor = BlockCursor::new_fileblockreader(&block_reader);
      89            0 :     for (k, v) in all {
      90            0 :         let value = cursor.read_blob(v.pos(), ctx).await?;
      91            0 :         println!("key:{} value_len:{}", k, value.len());
      92              :     }
      93              :     // TODO(chi): special handling for last key?
      94            0 :     Ok(())
      95            0 : }
      96              : 
      97            0 : pub(crate) async fn main(cmd: &LayerCmd) -> Result<()> {
      98            0 :     let ctx = RequestContext::new(TaskKind::DebugTool, DownloadBehavior::Error);
      99            0 :     match cmd {
     100            0 :         LayerCmd::List { path } => {
     101            0 :             for tenant in fs::read_dir(path.join(TENANTS_SEGMENT_NAME))? {
     102            0 :                 let tenant = tenant?;
     103            0 :                 if !tenant.file_type()?.is_dir() {
     104            0 :                     continue;
     105            0 :                 }
     106            0 :                 println!("tenant {}", tenant.file_name().to_string_lossy());
     107            0 :                 for timeline in fs::read_dir(tenant.path().join(TIMELINES_SEGMENT_NAME))? {
     108            0 :                     let timeline = timeline?;
     109            0 :                     if !timeline.file_type()?.is_dir() {
     110            0 :                         continue;
     111            0 :                     }
     112            0 :                     println!("- timeline {}", timeline.file_name().to_string_lossy());
     113              :                 }
     114              :             }
     115            0 :             Ok(())
     116              :         }
     117              :         LayerCmd::ListLayer {
     118            0 :             path,
     119            0 :             tenant,
     120            0 :             timeline,
     121            0 :         } => {
     122            0 :             let timeline_path = path
     123            0 :                 .join(TENANTS_SEGMENT_NAME)
     124            0 :                 .join(tenant)
     125            0 :                 .join(TIMELINES_SEGMENT_NAME)
     126            0 :                 .join(timeline);
     127            0 :             let mut idx = 0;
     128            0 :             for layer in fs::read_dir(timeline_path)? {
     129            0 :                 let layer = layer?;
     130            0 :                 if let Some(layer_file) = parse_filename(&layer.file_name().into_string().unwrap())
     131            0 :                 {
     132            0 :                     println!(
     133            0 :                         "[{:3}]  key:{}-{}\n       lsn:{}-{}\n       delta:{}",
     134            0 :                         idx,
     135            0 :                         layer_file.key_range.start,
     136            0 :                         layer_file.key_range.end,
     137            0 :                         layer_file.lsn_range.start,
     138            0 :                         layer_file.lsn_range.end,
     139            0 :                         layer_file.is_delta,
     140            0 :                     );
     141            0 :                     idx += 1;
     142            0 :                 }
     143              :             }
     144            0 :             Ok(())
     145              :         }
     146              :         LayerCmd::DumpLayer {
     147            0 :             path,
     148            0 :             tenant,
     149            0 :             timeline,
     150            0 :             id,
     151            0 :         } => {
     152            0 :             let timeline_path = path
     153            0 :                 .join("tenants")
     154            0 :                 .join(tenant)
     155            0 :                 .join("timelines")
     156            0 :                 .join(timeline);
     157            0 :             let mut idx = 0;
     158            0 :             for layer in fs::read_dir(timeline_path)? {
     159            0 :                 let layer = layer?;
     160            0 :                 if let Some(layer_file) = parse_filename(&layer.file_name().into_string().unwrap())
     161              :                 {
     162            0 :                     if *id == idx {
     163              :                         // TODO(chi): dedup code
     164            0 :                         println!(
     165            0 :                             "[{:3}]  key:{}-{}\n       lsn:{}-{}\n       delta:{}",
     166            0 :                             idx,
     167            0 :                             layer_file.key_range.start,
     168            0 :                             layer_file.key_range.end,
     169            0 :                             layer_file.lsn_range.start,
     170            0 :                             layer_file.lsn_range.end,
     171            0 :                             layer_file.is_delta,
     172            0 :                         );
     173            0 : 
     174            0 :                         if layer_file.is_delta {
     175            0 :                             read_delta_file(layer.path(), &ctx).await?;
     176              :                         } else {
     177            0 :                             anyhow::bail!("not supported yet :(");
     178              :                         }
     179              : 
     180            0 :                         break;
     181            0 :                     }
     182            0 :                     idx += 1;
     183            0 :                 }
     184              :             }
     185            0 :             Ok(())
     186              :         }
     187              :         LayerCmd::RewriteSummary {
     188            0 :             layer_file_path,
     189            0 :             new_tenant_id,
     190            0 :             new_timeline_id,
     191            0 :         } => {
     192            0 :             pageserver::virtual_file::init(10, virtual_file::api::IoEngineKind::StdFs);
     193            0 :             pageserver::page_cache::init(100);
     194            0 : 
     195            0 :             let ctx = RequestContext::new(TaskKind::DebugTool, DownloadBehavior::Error);
     196              : 
     197              :             macro_rules! rewrite_closure {
     198              :                 ($($summary_ty:tt)*) => {{
     199              :                     |summary| $($summary_ty)* {
     200              :                         tenant_id: new_tenant_id.unwrap_or(summary.tenant_id),
     201              :                         timeline_id: new_timeline_id.unwrap_or(summary.timeline_id),
     202              :                         ..summary
     203            0 :                     }
     204              :                 }};
     205              :             }
     206              : 
     207            0 :             let res = ImageLayer::rewrite_summary(
     208            0 :                 layer_file_path,
     209            0 :                 rewrite_closure!(image_layer::Summary),
     210            0 :                 &ctx,
     211            0 :             )
     212            0 :             .await;
     213            0 :             match res {
     214              :                 Ok(()) => {
     215            0 :                     println!("Successfully rewrote summary of image layer {layer_file_path}");
     216            0 :                     return Ok(());
     217              :                 }
     218            0 :                 Err(image_layer::RewriteSummaryError::MagicMismatch) => (), // fallthrough
     219            0 :                 Err(image_layer::RewriteSummaryError::Other(e)) => {
     220            0 :                     return Err(e);
     221              :                 }
     222              :             }
     223              : 
     224            0 :             let res = DeltaLayer::rewrite_summary(
     225            0 :                 layer_file_path,
     226            0 :                 rewrite_closure!(delta_layer::Summary),
     227            0 :                 &ctx,
     228            0 :             )
     229            0 :             .await;
     230            0 :             match res {
     231              :                 Ok(()) => {
     232            0 :                     println!("Successfully rewrote summary of delta layer {layer_file_path}");
     233            0 :                     return Ok(());
     234              :                 }
     235            0 :                 Err(delta_layer::RewriteSummaryError::MagicMismatch) => (), // fallthrough
     236            0 :                 Err(delta_layer::RewriteSummaryError::Other(e)) => {
     237            0 :                     return Err(e);
     238              :                 }
     239              :             }
     240              : 
     241            0 :             anyhow::bail!("not an image or delta layer: {layer_file_path}");
     242              :         }
     243              :     }
     244            0 : }
        

Generated by: LCOV version 2.1-beta