diff options
| author | nsfisis <nsfisis@gmail.com> | 2026-05-15 23:27:03 +0900 |
|---|---|---|
| committer | nsfisis <nsfisis@gmail.com> | 2026-05-16 10:00:40 +0900 |
| commit | 35df88af4fe7930c52ecd6c54683d71d823a5e7f (patch) | |
| tree | 9052843de4806668e200296004dd64c0360e632e /crates/shirabe | |
| parent | d05d73dd8039b2858c5f715eec2d28c58b2a33d4 (diff) | |
| download | php-shirabe-35df88af4fe7930c52ecd6c54683d71d823a5e7f.tar.gz php-shirabe-35df88af4fe7930c52ecd6c54683d71d823a5e7f.tar.zst php-shirabe-35df88af4fe7930c52ecd6c54683d71d823a5e7f.zip | |
feat(port): port ExecCommand.php
Diffstat (limited to 'crates/shirabe')
| -rw-r--r-- | crates/shirabe/src/command/exec_command.rs | 141 |
1 files changed, 141 insertions, 0 deletions
diff --git a/crates/shirabe/src/command/exec_command.rs b/crates/shirabe/src/command/exec_command.rs index f5e884f..432e3a0 100644 --- a/crates/shirabe/src/command/exec_command.rs +++ b/crates/shirabe/src/command/exec_command.rs @@ -1 +1,142 @@ //! ref: composer/src/Composer/Command/ExecCommand.php + +use anyhow::Result; +use shirabe_external_packages::symfony::console::input::input_interface::InputInterface; +use shirabe_external_packages::symfony::console::output::output_interface::OutputInterface; +use shirabe_php_shim::{basename, chdir, getcwd, glob, PhpMixed, RuntimeException}; + +use crate::command::base_command::BaseCommand; +use crate::console::input::input_argument::InputArgument; +use crate::console::input::input_option::InputOption; + +#[derive(Debug)] +pub struct ExecCommand { + inner: BaseCommand, +} + +impl ExecCommand { + pub fn configure(&mut self) { + self.inner + .set_name("exec") + .set_description("Executes a vendored binary/script") + .set_definition(vec![ + InputOption::new("list", Some(PhpMixed::String("l".to_string())), Some(InputOption::VALUE_NONE), "", None, vec![]), + InputArgument::new( + "binary", + Some(InputArgument::OPTIONAL), + "The binary to run, e.g. phpunit", + None, + // suggestion callback deferred; binaries listed at runtime via get_binaries + vec![], + ), + InputArgument::new( + "args", + Some(InputArgument::IS_ARRAY | InputArgument::OPTIONAL), + "Arguments to pass to the binary. Use <info>--</info> to separate from composer arguments", + None, + vec![], + ), + ]) + .set_help( + "Executes a vendored binary/script.\n\n\ + Read more at https://getcomposer.org/doc/03-cli.md#exec" + ); + } + + pub fn interact(&self, input: &mut dyn InputInterface, _output: &dyn OutputInterface) -> Result<()> { + let binaries = self.get_binaries(false)?; + if binaries.is_empty() { + return Ok(()); + } + + if input.get_argument("binary").as_string_opt().is_some() || input.get_option("list").as_bool().unwrap_or(false) { + return Ok(()); + } + + let io = self.inner.get_io(); + let binary = io.select( + "Binary to run: ".to_string(), + binaries.clone(), + PhpMixed::String(String::new()), + PhpMixed::Int(1), + "Invalid binary name \"%s\"".to_string(), + false, + ); + + if let Some(idx) = binary.as_int() { + input.set_argument("binary", &binaries[idx as usize]); + } + + Ok(()) + } + + pub fn execute(&self, input: &dyn InputInterface, _output: &dyn OutputInterface) -> Result<i64> { + let composer = self.inner.require_composer()?; + + if input.get_option("list").as_bool().unwrap_or(false) || input.get_argument("binary").as_string_opt().is_none() { + let bins = self.get_binaries(true)?; + if bins.is_empty() { + let bin_dir = composer.get_config().get("bin-dir").as_string().unwrap_or("").to_string(); + return Err(RuntimeException { + message: format!("No binaries found in composer.json or in bin-dir ({})", bin_dir), + code: 0, + } + .into()); + } + + self.inner.get_io().write("<comment>Available binaries:</comment>"); + for bin in &bins { + self.inner.get_io().write(&format!("<info>- {}</info>", bin)); + } + + return Ok(0); + } + + let binary = input.get_argument("binary").as_string().unwrap_or("").to_string(); + + let dispatcher = composer.get_event_dispatcher(); + dispatcher.add_listener("__exec_command", &binary); + + let initial_working_directory = self.inner.get_application().get_initial_working_directory(); + if let Some(ref iwd) = initial_working_directory { + if getcwd().as_deref() != Some(iwd.as_str()) { + chdir(iwd).map_err(|e| { + RuntimeException { message: format!("Could not switch back to working directory \"{}\"", iwd.display()), code: 0 } + })?; + } + } + + let args = input.get_argument("args") + .as_list() + .map(|l| l.iter().filter_map(|v| v.as_string().map(|s| s.to_string())).collect::<Vec<_>>()) + .unwrap_or_default(); + + Ok(dispatcher.dispatch_script("__exec_command", true, args)?) + } + + fn get_binaries(&self, for_display: bool) -> Result<Vec<String>> { + let composer = self.inner.require_composer()?; + let bin_dir = composer.get_config().get("bin-dir").as_string().unwrap_or("").to_string(); + let bins = glob(&format!("{}/*", bin_dir)); + let local_bins_raw: Vec<String> = composer.get_package().get_binaries(); + let local_bins: Vec<String> = if for_display { + local_bins_raw.into_iter().map(|e| format!("{} (local)", e)).collect() + } else { + local_bins_raw + }; + + let mut binaries: Vec<String> = Vec::new(); + let mut previous_bin: Option<String> = None; + for bin in bins.iter().chain(local_bins.iter()) { + if let Some(prev) = &previous_bin { + if bin == &format!("{}.bat", prev) { + continue; + } + } + previous_bin = Some(bin.clone()); + binaries.push(basename(bin)); + } + + Ok(binaries) + } +} |
