summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDirk-Jan C. Binnema <djcb@djcbsoftware.nl>2025-11-25 21:25:21 +0200
committerDirk-Jan C. Binnema <djcb@djcbsoftware.nl>2025-11-25 21:28:02 +0200
commitfc4d5b01a703e8c8cc390cfea135f08d3b45ccab (patch)
tree79f813eaad8c8c6318e6f4b35577f8d2afcb9e9e
parentfb806392c35d7a63258c91badef56de55b0daafe (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.cc54
-rw-r--r--lib/mu-maildir.cc6
-rw-r--r--lib/tests/bench-indexer.cc13
-rw-r--r--mu/mu-options.cc7
-rw-r--r--mu/tests/test-mu-query.cc20
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);