#![allow(unused_imports)] #![allow(dead_code)] use std::io::Write; #[doc(hidden)] pub mod res; mod splitter; use crate::error::{ProgramInternalExecuteError, ProgramPanic}; use crate::program::repl_exec::splitter::split_input_string; use crate::{Program, ProgramCollect, RenderResult}; use crate::{program::repl_exec::res::REPL, this}; #[cfg(not(feature = "async"))] impl Program where C: ProgramCollect + Send + Sync + 'static, { /// Executes the REPL interactive CLI mode. /// /// This method starts an infinite loop that continuously reads user input, parses commands, executes them, /// and displays the execution result or error message. It is suitable for scenarios requiring command-line interaction with the user. pub fn exec_repl(mut self) { // Inject default REPL resource self.with_resource(REPL::default()); self.run_hook_repl_on_begin(); self.exec_wrapper(|p| -> () { loop { p.run_hook_repl_pre_readline(); let readline = p.run_hook_repl_readline().unwrap_or_default(); p.run_hook_repl_post_readline(&readline); let args = split_input_string(readline.clone()); p.run_hook_repl_pre_exec(&args); match exec_once(p, args) { Ok(r) => { p.run_hook_repl_on_receive_result(&r); } Err(ProgramInternalExecuteError::REPLPanic(panic)) => { p.run_hook_repl_on_panic(&panic); } _ => {} } p.run_hook_repl_post_exec(); if this::().res::().unwrap().exit { p.run_hook_repl_exit(); break; } p.run_hook_repl_loop_once(); } }); } } #[cfg(feature = "async")] impl Program where C: ProgramCollect + Send + Sync + 'static, { /// Executes the REPL interactive CLI mode. /// /// This method starts an infinite loop that continuously reads user input, parses commands, executes them, /// and displays the execution result or error message. It is suitable for scenarios requiring command-line interaction with the user. /// /// **Note:** When the `async` feature is enabled, panic unwinding is not supported. /// Any panics during command execution will result in an abort rather than being caught and handled gracefully. pub async fn exec_repl(self) { // Inject default REPL resource self.with_resource(REPL::default()); self.run_hook_repl_on_begin(); self.exec_wrapper(async |p| -> () { loop { p.run_hook_repl_pre_readline(); let readline = p.run_hook_repl_readline().unwrap_or_default(); p.run_hook_repl_post_readline(&readline); let args = split_input_string(readline.clone()); p.run_hook_repl_pre_exec(&args); match exec_once(p, args).await { Ok(r) => { p.run_hook_repl_on_receive_result(&r); } _ => {} } p.run_hook_repl_post_exec(); if this::().res::().unwrap().exit { p.run_hook_repl_exit(); break; } p.run_hook_repl_loop_once(); } }) .await; } } #[cfg(not(feature = "async"))] fn exec_once( p: &'static Program, args: Vec, ) -> Result where C: ProgramCollect + Send + Sync + 'static, { #[cfg(panic = "abort")] let exec_result = super::exec::exec_with_args(p, args); #[cfg(not(panic = "abort"))] let exec_result = { let exec_unwind_result = std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| { super::exec::exec_with_args(p, args) })); match exec_unwind_result { Err(panic_info) => { let panic_payload = ProgramPanic { payload: panic_info, }; let program = crate::program::THIS_PROGRAM .get() .unwrap() .as_ref() .unwrap() .downcast_ref::>() .unwrap(); program.run_hook_repl_on_panic(&panic_payload); Err(ProgramInternalExecuteError::REPLPanic(panic_payload)) } Ok(r) => r, } }; exec_result } #[cfg(feature = "async")] async fn exec_once( p: &'static Program, args: Vec, ) -> Result where C: ProgramCollect + Send + Sync + 'static, { super::exec::exec_with_args(p, args).await }