| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  | use std::collections::HashMap;
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  | use std::sync::Arc;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use clap::Parser;
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  | use mini_lsm::compact::{
 | 
					
						
							|  |  |  |     SimpleLeveledCompactionController, SimpleLeveledCompactionOptions, TieredCompactionController,
 | 
					
						
							|  |  |  |     TieredCompactionOptions,
 | 
					
						
							|  |  |  | };
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  | use mini_lsm::lsm_storage::LsmStorageInner;
 | 
					
						
							|  |  |  | use mini_lsm::mem_table::MemTable;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Parser, Debug)]
 | 
					
						
							|  |  |  | #[command(author, version, about, long_about = None)]
 | 
					
						
							|  |  |  | enum Args {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |     Simple {
 | 
					
						
							|  |  |  |         #[clap(long)]
 | 
					
						
							|  |  |  |         dump_real_id: bool,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "2")]
 | 
					
						
							|  |  |  |         level0_file_num_compaction_trigger: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "3")]
 | 
					
						
							|  |  |  |         max_levels: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "200")]
 | 
					
						
							|  |  |  |         size_ratio_percent: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "50")]
 | 
					
						
							|  |  |  |         iterations: usize,
 | 
					
						
							|  |  |  |     },
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |     Tiered {
 | 
					
						
							|  |  |  |         #[clap(long)]
 | 
					
						
							|  |  |  |         dump_real_id: bool,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "3")]
 | 
					
						
							|  |  |  |         level0_file_num_compaction_trigger: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "200")]
 | 
					
						
							|  |  |  |         max_size_amplification_percent: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "1")]
 | 
					
						
							|  |  |  |         size_ratio: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "2")]
 | 
					
						
							|  |  |  |         min_merge_width: usize,
 | 
					
						
							|  |  |  |         #[clap(long, default_value = "50")]
 | 
					
						
							|  |  |  |         iterations: usize,
 | 
					
						
							|  |  |  |     },
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |     Leveled {},
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct MockStorage {
 | 
					
						
							|  |  |  |     snapshot: LsmStorageInner,
 | 
					
						
							|  |  |  |     next_sst_id: usize,
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |     /// Maps SST ID to the original flushed SST ID
 | 
					
						
							|  |  |  |     file_list: HashMap<usize, usize>,
 | 
					
						
							|  |  |  |     total_flushes: usize,
 | 
					
						
							|  |  |  |     total_writes: usize,
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl MockStorage {
 | 
					
						
							|  |  |  |     pub fn new() -> Self {
 | 
					
						
							|  |  |  |         let snapshot = LsmStorageInner {
 | 
					
						
							|  |  |  |             memtable: Arc::new(MemTable::create()),
 | 
					
						
							|  |  |  |             imm_memtables: Vec::new(),
 | 
					
						
							|  |  |  |             l0_sstables: Vec::new(),
 | 
					
						
							|  |  |  |             levels: Vec::new(),
 | 
					
						
							|  |  |  |             sstables: Default::default(),
 | 
					
						
							|  |  |  |         };
 | 
					
						
							|  |  |  |         Self {
 | 
					
						
							|  |  |  |             snapshot,
 | 
					
						
							|  |  |  |             next_sst_id: 0,
 | 
					
						
							|  |  |  |             file_list: Default::default(),
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |             total_flushes: 0,
 | 
					
						
							|  |  |  |             total_writes: 0,
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn generate_sst_id(&mut self) -> usize {
 | 
					
						
							|  |  |  |         let id = self.next_sst_id;
 | 
					
						
							|  |  |  |         self.next_sst_id += 1;
 | 
					
						
							|  |  |  |         id
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |     pub fn flush_sst_to_l0(&mut self) {
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |         let id = self.generate_sst_id();
 | 
					
						
							|  |  |  |         self.snapshot.l0_sstables.push(id);
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |         self.file_list.insert(id, id);
 | 
					
						
							|  |  |  |         self.total_flushes += 1;
 | 
					
						
							|  |  |  |         self.total_writes += 1;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn flush_sst_to_new_tier(&mut self) {
 | 
					
						
							|  |  |  |         let id = self.generate_sst_id();
 | 
					
						
							|  |  |  |         self.snapshot.levels.insert(0, (id, vec![id]));
 | 
					
						
							|  |  |  |         self.file_list.insert(id, id);
 | 
					
						
							|  |  |  |         self.total_flushes += 1;
 | 
					
						
							|  |  |  |         self.total_writes += 1;
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn remove(&mut self, files_to_remove: &[usize]) {
 | 
					
						
							|  |  |  |         for file_id in files_to_remove {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |             let ret = self.file_list.remove(file_id);
 | 
					
						
							|  |  |  |             assert!(ret.is_some(), "failed to remove file {}", file_id);
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |     pub fn dump_original_id(&self, always_show_l0: bool) {
 | 
					
						
							|  |  |  |         if !self.snapshot.l0_sstables.is_empty() || always_show_l0 {
 | 
					
						
							|  |  |  |             println!(
 | 
					
						
							|  |  |  |                 "L0 ({}): {:?}",
 | 
					
						
							|  |  |  |                 self.snapshot.l0_sstables.len(),
 | 
					
						
							|  |  |  |                 self.snapshot.l0_sstables,
 | 
					
						
							|  |  |  |             );
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         for (level, files) in &self.snapshot.levels {
 | 
					
						
							|  |  |  |             println!(
 | 
					
						
							|  |  |  |                 "L{level} ({}): {:?}",
 | 
					
						
							|  |  |  |                 files.len(),
 | 
					
						
							|  |  |  |                 files.iter().map(|x| self.file_list[x]).collect::<Vec<_>>()
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |     pub fn dump_real_id(&self, always_show_l0: bool) {
 | 
					
						
							|  |  |  |         if !self.snapshot.l0_sstables.is_empty() || always_show_l0 {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |             println!(
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                 "L0 ({}): {:?}",
 | 
					
						
							|  |  |  |                 self.snapshot.l0_sstables.len(),
 | 
					
						
							|  |  |  |                 self.snapshot.l0_sstables,
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |             );
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |         for (level, files) in &self.snapshot.levels {
 | 
					
						
							|  |  |  |             println!("L{level} ({}): {:?}", files.len(), files);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fn main() {
 | 
					
						
							|  |  |  |     let args = Args::parse();
 | 
					
						
							|  |  |  |     match args {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |         Args::Simple {
 | 
					
						
							|  |  |  |             dump_real_id,
 | 
					
						
							|  |  |  |             size_ratio_percent,
 | 
					
						
							|  |  |  |             iterations,
 | 
					
						
							|  |  |  |             level0_file_num_compaction_trigger,
 | 
					
						
							|  |  |  |             max_levels,
 | 
					
						
							|  |  |  |         } => {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:49:43 +08:00
										 |  |  |             let controller =
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                 SimpleLeveledCompactionController::new(SimpleLeveledCompactionOptions {
 | 
					
						
							|  |  |  |                     size_ratio_percent,
 | 
					
						
							|  |  |  |                     level0_file_num_compaction_trigger,
 | 
					
						
							|  |  |  |                     max_levels,
 | 
					
						
							|  |  |  |                 });
 | 
					
						
							|  |  |  |             let mut storage = MockStorage::new();
 | 
					
						
							|  |  |  |             for i in 0..max_levels {
 | 
					
						
							|  |  |  |                 storage.snapshot.levels.push((i + 1, Vec::new()));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             let mut max_space = 0;
 | 
					
						
							|  |  |  |             for i in 0..iterations {
 | 
					
						
							|  |  |  |                 println!("=== Iteration {i} ===");
 | 
					
						
							|  |  |  |                 storage.flush_sst_to_l0();
 | 
					
						
							|  |  |  |                 println!("--- After Flush ---");
 | 
					
						
							|  |  |  |                 if dump_real_id {
 | 
					
						
							|  |  |  |                     storage.dump_real_id(true);
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     storage.dump_original_id(true);
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 let mut num_compactions = 0;
 | 
					
						
							|  |  |  |                 while let Some(task) = controller.generate_compaction_task(&storage.snapshot) {
 | 
					
						
							|  |  |  |                     println!("--- Compaction Task ---");
 | 
					
						
							|  |  |  |                     let mut sst_ids = Vec::new();
 | 
					
						
							|  |  |  |                     for file in task
 | 
					
						
							|  |  |  |                         .upper_level_sst_ids
 | 
					
						
							|  |  |  |                         .iter()
 | 
					
						
							|  |  |  |                         .chain(task.lower_level_sst_ids.iter())
 | 
					
						
							|  |  |  |                     {
 | 
					
						
							|  |  |  |                         let new_sst_id = storage.generate_sst_id();
 | 
					
						
							|  |  |  |                         sst_ids.push(new_sst_id);
 | 
					
						
							|  |  |  |                         storage.file_list.insert(new_sst_id, *file);
 | 
					
						
							|  |  |  |                         storage.total_writes += 1;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     print!(
 | 
					
						
							|  |  |  |                         "Upper L{} {:?} ",
 | 
					
						
							|  |  |  |                         task.upper_level.unwrap_or_default(),
 | 
					
						
							|  |  |  |                         task.upper_level_sst_ids
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							|  |  |  |                     print!(
 | 
					
						
							|  |  |  |                         "Lower L{} {:?} ",
 | 
					
						
							|  |  |  |                         task.lower_level, task.lower_level_sst_ids
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							|  |  |  |                     println!("-> {:?}", sst_ids);
 | 
					
						
							|  |  |  |                     max_space = max_space.max(storage.file_list.len());
 | 
					
						
							|  |  |  |                     let (snapshot, del) =
 | 
					
						
							|  |  |  |                         controller.apply_compaction_result(&storage.snapshot, &task, &sst_ids);
 | 
					
						
							|  |  |  |                     storage.snapshot = snapshot;
 | 
					
						
							|  |  |  |                     storage.remove(&del);
 | 
					
						
							|  |  |  |                     println!("--- After Compaction ---");
 | 
					
						
							|  |  |  |                     if dump_real_id {
 | 
					
						
							|  |  |  |                         storage.dump_real_id(true);
 | 
					
						
							|  |  |  |                     } else {
 | 
					
						
							|  |  |  |                         storage.dump_original_id(true);
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							| 
									
										
										
										
											2024-01-17 16:23:58 +08:00
										 |  |  |                     num_compactions += 1;
 | 
					
						
							|  |  |  |                     if num_compactions >= max_levels * 2 {
 | 
					
						
							|  |  |  |                         panic!("compaction does not converge?");
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                 }
 | 
					
						
							|  |  |  |                 if num_compactions == 0 {
 | 
					
						
							|  |  |  |                     println!("no compaction triggered");
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     println!("{num_compactions} compaction triggered in this iteration");
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 max_space = max_space.max(storage.file_list.len());
 | 
					
						
							|  |  |  |                 println!("--- Statistics ---");
 | 
					
						
							|  |  |  |                 println!(
 | 
					
						
							|  |  |  |                     "Write Amplification: {}/{}={:.3}x",
 | 
					
						
							|  |  |  |                     storage.total_writes,
 | 
					
						
							|  |  |  |                     storage.total_flushes,
 | 
					
						
							|  |  |  |                     storage.total_writes as f64 / storage.total_flushes as f64
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							|  |  |  |                 println!(
 | 
					
						
							|  |  |  |                     "Space Amplification: {}/{}={:.3}x",
 | 
					
						
							|  |  |  |                     max_space,
 | 
					
						
							|  |  |  |                     storage.total_flushes,
 | 
					
						
							|  |  |  |                     max_space as f64 / storage.total_flushes as f64
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							|  |  |  |                 println!(
 | 
					
						
							|  |  |  |                     "Read Amplification: {}x",
 | 
					
						
							|  |  |  |                     storage.snapshot.l0_sstables.len()
 | 
					
						
							|  |  |  |                         + storage
 | 
					
						
							|  |  |  |                             .snapshot
 | 
					
						
							|  |  |  |                             .levels
 | 
					
						
							|  |  |  |                             .iter()
 | 
					
						
							|  |  |  |                             .filter(|(_, f)| !f.is_empty())
 | 
					
						
							|  |  |  |                             .count()
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							|  |  |  |                 println!();
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |         Args::Tiered {
 | 
					
						
							|  |  |  |             dump_real_id,
 | 
					
						
							|  |  |  |             level0_file_num_compaction_trigger,
 | 
					
						
							|  |  |  |             max_size_amplification_percent,
 | 
					
						
							|  |  |  |             size_ratio,
 | 
					
						
							|  |  |  |             min_merge_width,
 | 
					
						
							|  |  |  |             iterations,
 | 
					
						
							|  |  |  |         } => {
 | 
					
						
							|  |  |  |             let controller = TieredCompactionController::new(TieredCompactionOptions {
 | 
					
						
							|  |  |  |                 level0_file_num_compaction_trigger,
 | 
					
						
							|  |  |  |                 max_size_amplification_percent,
 | 
					
						
							|  |  |  |                 size_ratio,
 | 
					
						
							|  |  |  |                 min_merge_width,
 | 
					
						
							|  |  |  |             });
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |             let mut storage = MockStorage::new();
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |             let mut max_space = 0;
 | 
					
						
							|  |  |  |             for i in 0..iterations {
 | 
					
						
							|  |  |  |                 println!("=== Iteration {i} ===");
 | 
					
						
							|  |  |  |                 storage.flush_sst_to_new_tier();
 | 
					
						
							|  |  |  |                 println!("--- After Flush ---");
 | 
					
						
							|  |  |  |                 if dump_real_id {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                     storage.dump_real_id(false);
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |                 } else {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                     storage.dump_original_id(false);
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |                 let task = controller.generate_compaction_task(&storage.snapshot);
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |                 println!("--- Compaction Task ---");
 | 
					
						
							|  |  |  |                 if let Some(task) = task {
 | 
					
						
							|  |  |  |                     let mut sst_ids = Vec::new();
 | 
					
						
							|  |  |  |                     for (tier_id, files) in &task.tiers {
 | 
					
						
							|  |  |  |                         for file in files {
 | 
					
						
							|  |  |  |                             let new_sst_id = storage.generate_sst_id();
 | 
					
						
							|  |  |  |                             sst_ids.push(new_sst_id);
 | 
					
						
							|  |  |  |                             storage.file_list.insert(new_sst_id, *file);
 | 
					
						
							|  |  |  |                             storage.total_writes += 1;
 | 
					
						
							|  |  |  |                         }
 | 
					
						
							|  |  |  |                         print!("L{} {:?} ", tier_id, files);
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     println!("-> {:?}", sst_ids);
 | 
					
						
							|  |  |  |                     max_space = max_space.max(storage.file_list.len());
 | 
					
						
							|  |  |  |                     let (snapshot, del) =
 | 
					
						
							|  |  |  |                         controller.apply_compaction_result(&storage.snapshot, &task, &sst_ids);
 | 
					
						
							|  |  |  |                     storage.snapshot = snapshot;
 | 
					
						
							|  |  |  |                     storage.remove(&del);
 | 
					
						
							|  |  |  |                     println!("--- After Compaction ---");
 | 
					
						
							|  |  |  |                     if dump_real_id {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                         storage.dump_real_id(false);
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |                     } else {
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:42:33 +08:00
										 |  |  |                         storage.dump_original_id(false);
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |                     }
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     println!("no compaction triggered");
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 max_space = max_space.max(storage.file_list.len());
 | 
					
						
							|  |  |  |                 println!("--- Statistics ---");
 | 
					
						
							|  |  |  |                 println!(
 | 
					
						
							|  |  |  |                     "Write Amplification: {}/{}={:.3}x",
 | 
					
						
							|  |  |  |                     storage.total_writes,
 | 
					
						
							|  |  |  |                     storage.total_flushes,
 | 
					
						
							|  |  |  |                     storage.total_writes as f64 / storage.total_flushes as f64
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							|  |  |  |                 println!(
 | 
					
						
							|  |  |  |                     "Space Amplification: {}/{}={:.3}x",
 | 
					
						
							|  |  |  |                     max_space,
 | 
					
						
							|  |  |  |                     storage.total_flushes,
 | 
					
						
							|  |  |  |                     max_space as f64 / storage.total_flushes as f64
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							| 
									
										
										
										
											2024-01-17 15:49:43 +08:00
										 |  |  |                 println!(
 | 
					
						
							|  |  |  |                     "Read Amplification: {}x",
 | 
					
						
							|  |  |  |                     storage.snapshot.l0_sstables.len()
 | 
					
						
							|  |  |  |                         + storage
 | 
					
						
							|  |  |  |                             .snapshot
 | 
					
						
							|  |  |  |                             .levels
 | 
					
						
							|  |  |  |                             .iter()
 | 
					
						
							|  |  |  |                             .filter(|(_, f)| !f.is_empty())
 | 
					
						
							|  |  |  |                             .count()
 | 
					
						
							|  |  |  |                 );
 | 
					
						
							| 
									
										
										
										
											2024-01-17 14:51:15 +08:00
										 |  |  |                 println!();
 | 
					
						
							| 
									
										
										
										
											2024-01-16 16:30:01 +08:00
										 |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         Args::Leveled {} => {}
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 |