diff options
| author | Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> | 2025-11-25 21:25:21 +0200 |
|---|---|---|
| committer | Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> | 2025-11-25 21:28:02 +0200 |
| commit | fc4d5b01a703e8c8cc390cfea135f08d3b45ccab (patch) | |
| tree | 79f813eaad8c8c6318e6f4b35577f8d2afcb9e9e | |
| parent | fb806392c35d7a63258c91badef56de55b0daafe (diff) | |
improve invoking external commands
- don't make assumptions on where programs live (i.e., /bin/sh, /bin/rm,
/bin/mv) are not universal
- dont invoke shell when unnecessary
- improve error-handling
| -rw-r--r-- | guile/tests/test-mu-guile.cc | 54 | ||||
| -rw-r--r-- | lib/mu-maildir.cc | 6 | ||||
| -rw-r--r-- | lib/tests/bench-indexer.cc | 13 | ||||
| -rw-r--r-- | mu/mu-options.cc | 7 | ||||
| -rw-r--r-- | mu/tests/test-mu-query.cc | 20 |
5 files changed, 48 insertions, 52 deletions
diff --git a/guile/tests/test-mu-guile.cc b/guile/tests/test-mu-guile.cc index 09a53d0..5ec6894 100644 --- a/guile/tests/test-mu-guile.cc +++ b/guile/tests/test-mu-guile.cc @@ -1,5 +1,5 @@ /* -** Copyright (C) 2012-2023 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> +** Copyright (C) 2012-2025 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> ** ** This program is free software; you can redistribute it and/or modify it ** under the terms of the GNU General Public License as published by the @@ -28,6 +28,8 @@ #include <string.h> #include "utils/mu-test-utils.hh" +#include "utils/mu-utils-file.hh" + #include <lib/mu-store.hh> #include <utils/mu-utils.hh> @@ -38,26 +40,23 @@ static std::string test_dir; static std::string fill_database(void) { - const auto cmdline = mu_format( - "/bin/sh -c '" - "{} init --muhome={} --maildir={} --quiet; " - "{} index --muhome={} --quiet'", - MU_PROGRAM, - test_dir, - MU_TESTMAILDIR2, - MU_PROGRAM, - test_dir); - - if (g_test_verbose()) - mu_println("{}", cmdline); + { + const auto res = run_command0({MU_PROGRAM, + "--quiet", "init", + "--muhome", test_dir, + "--maildir", MU_TESTMAILDIR2}); + assert_valid_result(res); + } - GError *err{}; - if (!g_spawn_command_line_sync(cmdline.c_str(), NULL, NULL, NULL, &err)) { - mu_printerrln("Error: {}", err ? err->message : "?"); - g_clear_error(&err); - g_assert(0); + { + const auto res = run_command0({MU_PROGRAM, "--quiet", + "index", "--muhome", test_dir}); + assert_valid_result(res); } + if (g_test_verbose()) + mu_println("\nindexed {} @ {}", MU_TESTMAILDIR2, test_dir); + return test_dir; } @@ -72,22 +71,11 @@ test_something(const char* what) g_print("GUILE_LOAD_PATH: %s\n", GUILE_LOAD_PATH); const auto dir = fill_database(); - const auto cmdline = mu_format("{} -q -e main {}/test-mu-guile.scm " - "--muhome={} --test={}", - GUILE_BINARY, ABS_SRCDIR, - dir, what); + const auto res = run_command0({GUILE_BINARY, "-q", "-e", "main", + ABS_SRCDIR"/test-mu-guile.scm", + "--muhome", dir, "--test", what}); - if (g_test_verbose()) - mu_println("cmdline: {}", cmdline); - - GError *err{}; - int status{}; - if (!g_spawn_command_line_sync(cmdline.c_str(), NULL, NULL, &status, &err) || - status != 0) { - mu_printerrln("Error: {}", err ? err->message : "something went wrong"); - g_clear_error(&err); - g_assert(0); - } + assert_valid_result(res); } static void diff --git a/lib/mu-maildir.cc b/lib/mu-maildir.cc index 53152e5..e93566a 100644 --- a/lib/mu-maildir.cc +++ b/lib/mu-maildir.cc @@ -297,7 +297,11 @@ msg_move_g_file(const std::string& src, const std::string& dst) G_GNUC_UNUSED static Mu::Result<void> msg_move_mv_file(const std::string& src, const std::string& dst) { - if (auto res{run_command0({"/bin/mv", src, dst})}; !res) + static const auto mv_path{program_in_path("mv")}; + if (!mv_path) + return Err(Error::Code::File, "failed to find 'mv'"); + + if (auto res{run_command0({*mv_path, src, dst})}; !res) return Err(Error::Code::File, "error moving {}->{}; err={}", src, dst, res.error()); else return Ok(); diff --git a/lib/tests/bench-indexer.cc b/lib/tests/bench-indexer.cc index e57bcbd..2283be1 100644 --- a/lib/tests/bench-indexer.cc +++ b/lib/tests/bench-indexer.cc @@ -1,5 +1,5 @@ /* -** Copyright (C) 2022-2023 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> +** Copyright (C) 2022-2025 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> ** ** This program is free software; you can redistribute it and/or modify it ** under the terms of the GNU General Public License as published by the @@ -24,6 +24,8 @@ #include <fstream> #include <utils/mu-utils.hh> +#include <utils/mu-utils-file.hh> + #include <utils/mu-regex.hh> #include <mu-store.hh> #include "mu-maildir.hh" @@ -438,12 +440,9 @@ setup(const TestData& tdata) static void tear_down() { - /* ugly */ - GError *err{}; - const auto cmd{mu_format("/bin/rm -rf '{}' '{}'", BENCH_MAILDIRS, BENCH_STORE)}; - if (!g_spawn_command_line_sync(cmd.c_str(), NULL, NULL, NULL, &err)) { - mu_warning("error: {}", err ? err->message : "?"); - g_clear_error(&err); + for (auto&& dir : { BENCH_MAILDIRS, BENCH_STORE } ) { + if (const auto res = remove_directory(dir); !res) + mu_warning("failed to remove {}: {}", dir, res.error().what()); } } diff --git a/mu/mu-options.cc b/mu/mu-options.cc index 58e1385..ac9d885 100644 --- a/mu/mu-options.cc +++ b/mu/mu-options.cc @@ -829,13 +829,14 @@ add_scripts(CLI::App& app, Options& opts) static Result<Options> show_manpage(Options& opts, const std::string& name) { - char *path = g_find_program_in_path("man"); - if (!path) + const auto manprog{program_in_path("man")}; + if (!manprog) return Err(Error::Code::Command, "cannot find 'man' program"); GError* err{}; - auto cmd{to_string_gchar(std::move(path)) + " " + name}; + const auto cmd{mu_format("{} {}", *manprog, name)}; + // run_command0 doesn't work here. auto res = g_spawn_command_line_sync(cmd.c_str(), {}, {}, {}, &err); if (!res) return Err(Error::Code::Command, &err, diff --git a/mu/tests/test-mu-query.cc b/mu/tests/test-mu-query.cc index 09c0cde..5f04f86 100644 --- a/mu/tests/test-mu-query.cc +++ b/mu/tests/test-mu-query.cc @@ -1,5 +1,5 @@ /* -** Copyright (C) 2008-2022 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> +** Copyright (C) 2008-2025 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> ** ** This program is free software; you can redistribute it and/or modify it ** under the terms of the GNU General Public License as published by the @@ -47,16 +47,20 @@ make_database(const std::string& dbdir, const std::string& testdir) { /* use the env var rather than `--muhome` */ g_setenv("MUHOME", dbdir.c_str(), 1); - const auto cmdline{mu_format( - "/bin/sh -c '" - "{} --quiet init --maildir={} ; " - "{} --quiet index'", - MU_PROGRAM, testdir, MU_PROGRAM)}; + + { + const auto res = run_command0({MU_PROGRAM, "--quiet", "init", "--maildir", testdir}); + assert_valid_result(res); + } + + { + const auto res = run_command0({MU_PROGRAM, "--quiet", "index"}); + assert_valid_result(res); + } if (g_test_verbose()) - mu_printerrln("\n{}", cmdline); + mu_info("\nindexed {} @ {}", testdir, dbdir); - g_assert(g_spawn_command_line_sync(cmdline.c_str(), NULL, NULL, NULL, NULL)); auto xpath = join_paths(dbdir, "xapian"); /* ensure MUHOME worked */ g_assert_cmpuint(::access(xpath.c_str(), F_OK), ==, 0); |
