dotfiles_core/exec_wrapper.rs
1// Copyright (c) 2021-2022 Miguel Barreto and others
2//
3// Permission is hereby granted, free of charge, to any person obtaining
4// a copy of this software and associated documentation files (the
5// "Software"), to deal in the Software without restriction, including
6// without limitation the rights to use, copy, modify, merge, publish,
7// distribute, sublicense, and/or sell copies of the Software, and to
8// permit persons to whom the Software is furnished to do so, subject to
9// the following conditions:
10//
11// The above copyright notice and this permission notice shall be
12// included in all copies or substantial portions of the Software.
13//
14// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21
22//! Wraps some logic to run external commands and handle errors
23
24use std::io::Result as IoResult;
25use subprocess::{Exec, ExitStatus};
26
27
28use crate::error::{execution_error, process_until_first_err, DotfilesError};
29
30/// Executes the `cmd` and waits for it to finish.
31///
32/// If the execution returns a [std::io::Error] then it uses the `io_error_message`
33/// for the message in a DotfilesError.
34///
35/// If the execution finishes but in an error state, then it uses the
36/// `error_while_running_message` instead.
37pub fn execute_commands(
38 cmds: Vec<Exec>,
39 io_error_message: &str,
40 error_while_running_message: &str,
41) -> Result<(), DotfilesError> {
42 process_until_first_err(cmds.into_iter(), |cmd| {
43 handle_exec_error(cmd.join(),io_error_message, error_while_running_message)
44 })
45}
46
47fn handle_exec_error(
48 result: IoResult<ExitStatus>,
49 io_error_message: &str,
50 error_while_running_message: &str,
51) -> Result<(), DotfilesError> {
52 result.map_or_else(
53 |err| {
54 Err(DotfilesError::from(
55 io_error_message.into(),
56 execution_error(Some(err), None),
57 ))
58 },
59 |status| match status.success() {
60 true => Ok(()),
61 false => Err(DotfilesError::from(
62 format!("{error_while_running_message}, {status:?}"),
63 execution_error(None, Some(status)),
64 )),
65 },
66 )
67}