diff --git a/doc/man8/flux-imp.rst b/doc/man8/flux-imp.rst index c8e94084..0a199360 100644 --- a/doc/man8/flux-imp.rst +++ b/doc/man8/flux-imp.rst @@ -37,10 +37,6 @@ COMMANDS **exec** command configuration can be found in :man5:`flux-config-security-imp`. -**kill** - The **flux-imp kill** command is invoked by a multi-user instance to - send signals to jobs running as users other than the instance owner. - **run** The **flux-imp run** command is used by a Flux instance to execute arbitrary commands with privilege, typically a job prolog or epilog. diff --git a/src/imp/Makefile.am b/src/imp/Makefile.am index 31ac9761..42d170fa 100644 --- a/src/imp/Makefile.am +++ b/src/imp/Makefile.am @@ -62,7 +62,6 @@ IMP_SOURCES = \ signals.h \ cgroup.c \ cgroup.h \ - kill.c \ run.c \ exec/user.h \ exec/user.c \ diff --git a/src/imp/impcmd-list.c b/src/imp/impcmd-list.c index 231c1b32..af576f4b 100644 --- a/src/imp/impcmd-list.c +++ b/src/imp/impcmd-list.c @@ -18,8 +18,6 @@ extern int imp_casign_unprivileged (struct imp_state *imp, struct kv *); extern int imp_casign_privileged (struct imp_state *imp, struct kv *); extern int imp_exec_unprivileged (struct imp_state *imp, struct kv *); extern int imp_exec_privileged (struct imp_state *imp, struct kv *); -extern int imp_kill_unprivileged (struct imp_state *imp, struct kv *); -extern int imp_kill_privileged (struct imp_state *imp, struct kv *); extern int imp_run_unprivileged (struct imp_state *imp, struct kv *); extern int imp_run_privileged (struct imp_state *imp, struct kv *); @@ -41,9 +39,6 @@ struct impcmd impcmd_list[] = { { "exec", imp_exec_unprivileged, imp_exec_privileged }, - { "kill", - imp_kill_unprivileged, - imp_kill_privileged }, { "run", imp_run_unprivileged, imp_run_privileged }, diff --git a/src/imp/kill.c b/src/imp/kill.c deleted file mode 100644 index 5bdd0e14..00000000 --- a/src/imp/kill.c +++ /dev/null @@ -1,185 +0,0 @@ -/************************************************************\ - * Copyright 2020 Lawrence Livermore National Security, LLC - * (c.f. AUTHORS, NOTICE.LLNS, COPYING) - * - * This file is part of the Flux resource manager framework. - * For details, see https://github.com/flux-framework. - * - * SPDX-License-Identifier: LGPL-3.0 -\************************************************************/ - -/* flux-imp kill - signal tasks on behalf of requestor when authorized - * - * PURPOSE: - * - * Allow a non-privileged Flux instance to signal processes in - * jobs running as different users. - * - * OPERATION: - * - * The IMP kill command currently works under the assumption that the - * multiuser instance is running under systemd with Delegate=yes. This - * setting directs systemd to delegate ownership of the flux.service - * cgroup to the user under which Flux is runnng, e.g. "flux". - * - * Since all Flux jobs will be executed within this cgroup or a child, - * flux-imp kill may authorize signal delivery to any task where - * the task's cgroup is owned by the requesting user. - * - * These assumptions hold even if Flux is using a systemd user instance - * to launch multi-user jobs, since the systemd instance spawning jobs - * will also be running under a delegated cgroup. - * - */ - -#if HAVE_CONFIG_H -#include -#endif - -#include -#include -#include -#include -#include -#include -#include -#include - -#include -#include - -#include "src/libutil/kv.h" -#include "src/libutil/strlcpy.h" - -#include "imp_log.h" -#include "imp_state.h" -#include "impcmd.h" -#include "privsep.h" -#include "pidinfo.h" - -/* Return true if the user executing the IMP is allowed to run - * 'flux-imp kill'. This is the same set of users allowed to run - * 'flux-imp exec', so look in exec.allowed-users. - */ -static bool imp_kill_allowed (const cf_t *conf) -{ - struct passwd * pwd = getpwuid (getuid ()); - const cf_t *exec = cf_get_in (conf, "exec"); - - if (pwd && exec) - return cf_array_contains (cf_get_in (exec, "allowed-users"), - pwd->pw_name); - return false; -} - -static void check_and_kill_process (struct imp_state *imp, pid_t pid, int sig) -{ - uid_t user = getuid (); - struct pid_info *p = NULL; - - if (!imp_kill_allowed (imp->conf)) - imp_die (1, "kill command not allowed"); - - if (!(p = pid_info_create ((pid_t) pid))) - imp_die (1, "kill: failed to initialize pid info: %s", - strerror (errno)); - if (sig == 0) { - printf ("pid=%ju owner=%ju cg_path=%s cg_owner=%ju", - (uintmax_t) pid, - (uintmax_t) p->pid_owner, - p->cg_path, - (uintmax_t) p->cg_owner); - pid_info_destroy (p); - return; - } - - /* Check if pid is in pids cgroup owned by IMP user */ - if (p->cg_owner != user - && p->pid_owner != user) - imp_die (1, - "kill: refusing request from uid=%ju to kill pid %jd (owner=%ju)", - (uintmax_t) user, - (intmax_t) pid, - (uintmax_t) p->cg_owner); - - /* If pid is owned by root and is a flux-imp process, then deliver - * signal to child process instead (presumably a job shell) - */ - if (p->pid_owner == 0 - && strcmp (p->command, "flux-imp") == 0) { - int count = pid_kill_children (pid, sig); - if (count < 0) - imp_die (1, - "kill: failed to signal flux-imp children: %s", - strerror (errno)); - else if (count == 0) - imp_die (1, "kill: killed 0 flux-imp children"); - } - else if (kill (pid, sig) < 0) - imp_die (1, "kill: %jd sig=%ju: %s", - (intmax_t) pid, - (uintmax_t) sig, - strerror (errno)); - - pid_info_destroy (p); -} - - -/* Read pid and signal from the privsep pipe, then check if user - * is allowed to kill the target process. - */ -int imp_kill_privileged (struct imp_state *imp, struct kv *kv) -{ - int64_t pid; - int64_t signum; - - if (kv_get (kv, "pid", KV_INT64, &pid) < 0) - imp_die (1, "kill: failed to get pid"); - if (kv_get (kv, "signal", KV_INT64, &signum) < 0) - imp_die (1, "kill: failed to get signal"); - - check_and_kill_process (imp, pid, signum); - return 0; -} - -/* Unprivileged process reads signal and pid from cmdline and - * sends to parent over privsep pipe. If not running privileged, - * try killing as requesting user (used for testing). - */ -int imp_kill_unprivileged (struct imp_state *imp, struct kv *kv) -{ - char *p = NULL; - int64_t pid = 0; - int64_t signum = -1; - - if (imp->argc < 4) - imp_die (1, "kill: Usage flux-imp kill SIGNAL PID"); - - errno = 0; - if ((signum = strtol (imp->argv[2], &p, 10)) < 0 - || errno != 0 - || *p != '\0') - imp_die (1, "kill: invalid SIGNAL %s", imp->argv[2]); - - /* PID of 0 is explicitly forbidden here as it could be used - * to inadvertently kill our parent. - */ - if ((pid = strtol (imp->argv[3], &p, 10)) == 0 - || *p != '\0') - imp_die (1, "kill: invalid PID %s", imp->argv[3]); - - if (kv_put (kv, "pid", KV_INT64, pid) < 0) - imp_die (1, "kill: kv_put pid: %s", strerror (errno)); - if (kv_put (kv, "signal", KV_INT64, signum) < 0) - imp_die (1, "kill: kv_put signum: %s", strerror (errno)); - - if (!imp->ps) - check_and_kill_process (imp, pid, signum); - else if (privsep_write_kv (imp->ps, kv) < 0) - imp_die (1, "kill: failed to communicate with privsep parent"); - - return 0; -} - -/* vi: ts=4 sw=4 expandtab - */ diff --git a/src/imp/pidinfo.c b/src/imp/pidinfo.c index 10771f7e..60122b93 100644 --- a/src/imp/pidinfo.c +++ b/src/imp/pidinfo.c @@ -25,203 +25,12 @@ #include #include -#include -#ifdef HAVE_LINUX_MAGIC_H -#include -#endif -#ifndef TMPFS_MAGIC -#define TMPFS_MAGIC 0x01021994 /* from linux/magic.h */ -#endif -#ifndef CGROUP_SUPER_MAGIC -#define CGROUP_SUPER_MAGIC 0x27e0eb -#endif - #include #include -#include "src/libutil/strlcpy.h" - #include "pidinfo.h" #include "imp_log.h" -struct cgroup_info { - char mount_dir[PATH_MAX + 1]; - bool unified; -}; - -/* Determine if this system is using the unified (v2) or legacy (v1) - * cgroups hierarchy (See https://systemd.io/CGROUP_DELEGATION/) - * and mount point for systemd managed cgroups. - */ -static int cgroup_info_init (struct cgroup_info *cg) -{ - struct statfs fs; - - (void) strlcpy (cg->mount_dir, "/sys/fs/cgroup", sizeof (cg->mount_dir)); - cg->unified = true; - - if (statfs (cg->mount_dir, &fs) < 0) - return -1; - -#ifdef CGROUP2_SUPER_MAGIC - /* if cgroup2 fs mounted: unified hierarchy for all users of cgroupfs - */ - if (fs.f_type == CGROUP2_SUPER_MAGIC) - return 0; -#endif /* CGROUP2_SUPER_MAGIC */ - - /* O/w, if /sys/fs/cgroup is mounted as tmpfs, we need to check - * for /sys/fs/cgroup/systemd mounted as cgroupfs (legacy). - * We do not support hybrid mode (/sys/fs/cgroup/systemd or - * /sys/fs/cgroup/unified mounted as cgroup2fs), since there were - * no systems on which to test this configuration. - */ - if (fs.f_type == TMPFS_MAGIC) { - - (void) strlcpy (cg->mount_dir, - "/sys/fs/cgroup/systemd", - sizeof (cg->mount_dir)); - if (statfs (cg->mount_dir, &fs) == 0 - && fs.f_type == CGROUP_SUPER_MAGIC) { - cg->unified = false; - return 0; - } - } - - /* Unable to determine cgroup mount point and/or unified vs legacy */ - return -1; -} - - -/* Store the command name from /proc/PID/comm into buffer 'buf' of size 'len'. - */ -static int pid_command (pid_t pid, char *buf, int len) -{ - int rc = -1; - FILE *fp = NULL; - int n; - size_t size = 0; - char *line = NULL; - char file [64]; - int saved_errno; - - if (buf == NULL || len <= 0) { - errno = EINVAL; - return -1; - } - - /* 64 bytes is guaranteed to hold /proc/%ju/comm, assuming largest - * unsigned integer pid would be 21 characters (2^64-1) + 11 characters - * for "/proc/" + "/comm" + some slack. - */ - (void) snprintf (file, sizeof (file), "/proc/%ju/comm", (uintmax_t) pid); - - if (!(fp = fopen (file, "r"))) - return -1; - if (getline (&line, &size, fp) < 0) - goto out; - if ((n = strlen (line)) > len) { - errno = ENOSPC; - goto out; - } - /* - * Remove trailing newline and copy command into destination buffer. - * No need to check return code since size of destination was already - * checked above. - */ - if (line[n-1] == '\n') - line[n-1] = '\0'; - (void) strlcpy (buf, line, len); - - rc = 0; -out: - saved_errno = errno; - free (line); - fclose (fp); - errno = saved_errno; - return rc; -} - -/* - * Looks up the 'name=systemd'[*] subsystem relative cgroup path in - * /proc/PID/cgroups and prepends `cgroup_mount_dir` to get the - * full path. - * - * [*] perhaps could also use the "pids" cgroup. - */ -static int pid_systemd_cgroup_path (pid_t pid, char *buf, int len) -{ - int rc = -1; - FILE *fp; - size_t size = 0; - int n; - char file [64]; - char *line = NULL; - struct cgroup_info cgroup; - int saved_errno; - - if (cgroup_info_init (&cgroup) < 0) - return -1; - - /* 64 bytes is guaranteed to hold /proc/%ju/comm, assuming largest - * unsigned integer pid would be 21 characters (2^64-1) + 13 characters - * for "/proc/" + "/cgroup" + some slack. - */ - (void) snprintf (file, sizeof (file), "/proc/%ju/cgroup", (uintmax_t) pid); - if (!(fp = fopen (file, "r"))) - return -1; - - while ((n = getline (&line, &size, fp)) >= 0) { - char *nl; - char *relpath = NULL; - char *subsys = strchr (line, ':'); - if ((nl = strchr (line, '\n'))) - *nl = '\0'; - if (subsys == NULL || *(++subsys) == '\0' - || !(relpath = strchr (subsys, ':'))) - continue; - /* Nullify subsys, relpath is already nul-terminated at newline */ - *(relpath++) = '\0'; - if (cgroup.unified || strcmp (subsys, "name=systemd") == 0) { - n = snprintf (buf, len, "%s%s", cgroup.mount_dir, relpath); - if ((n > 0) && (n < len)) - rc = 0; - break; - } - } - - if (rc < 0) - errno = ENOENT; - - saved_errno = errno; - free (line); - fclose (fp); - errno = saved_errno; - return rc; -} - -/* return the file owner of 'path' - */ -static uid_t path_owner (const char *path) -{ - struct stat st; - if (stat (path, &st) < 0) - return ((uid_t) -1); - return st.st_uid; -} - -/* return the owner of pid. -1 on failure. - */ -static uid_t pid_owner (pid_t pid) -{ - char path [64]; - - /* /proc/%ju is guaranteed to fit in 64 bytes: - */ - (void) snprintf (path, sizeof (path), "/proc/%ju", (uintmax_t) pid); - return path_owner (path); -} - static int parse_pid (const char *s, pid_t *ppid) { unsigned long val; @@ -281,42 +90,6 @@ static pid_t pid_ppid (pid_t pid) return ppid; } -void pid_info_destroy (struct pid_info *pi) -{ - if (pi) { - int saved_errno = errno; - free (pi); - errno = saved_errno; - } -} - -struct pid_info *pid_info_create (pid_t pid) -{ - struct pid_info *pi; - - if (pid == 0) { - errno = EINVAL; - return NULL; - } - if (!(pi = calloc (1, sizeof (*pi)))) - return NULL; - if (pid < 0) - pid = -pid; - pi->pid = pid; - if ((pi->pid_owner = pid_owner (pid)) == (uid_t) -1) - goto err; - if (pid_systemd_cgroup_path (pid, pi->cg_path, sizeof (pi->cg_path)) < 0) - goto err; - if ((pi->cg_owner = path_owner (pi->cg_path)) == (uid_t) -1) - goto err; - if (pid_command (pid, pi->command, sizeof (pi->command)) < 0) - goto err; - return pi; -err: - pid_info_destroy (pi); - return NULL; -} - int pid_kill_children_fallback (pid_t parent, int sig) { int count = 0; diff --git a/src/imp/pidinfo.h b/src/imp/pidinfo.h index 57d73c01..ef273e2a 100644 --- a/src/imp/pidinfo.h +++ b/src/imp/pidinfo.h @@ -11,17 +11,6 @@ #ifndef HAVE_PIDINFO_H #define HAVE_PIDINFO_H 1 -struct pid_info { - pid_t pid; - char command [64]; - uid_t pid_owner; - char cg_path [4096]; - uid_t cg_owner; -}; - -struct pid_info *pid_info_create (pid_t pid); -void pid_info_destroy (struct pid_info *pi); - /* Send signal to any children of pid. * Returns the number of children signaled or -1 if an error occurred. */ diff --git a/src/imp/test/pidinfo.c b/src/imp/test/pidinfo.c index 74d021a4..84b1f894 100644 --- a/src/imp/test/pidinfo.c +++ b/src/imp/test/pidinfo.c @@ -149,33 +149,7 @@ static void pid_kill_tests (void) int main (void) { - struct pid_info *p; - - ok (pid_info_create (0) == NULL && errno == EINVAL, - "pid_info_create (0) fails with EINVAL"); - - ok ((p = pid_info_create (getpid ())) != NULL, - "pid_info_create (getpid ()) works"); - ok (p->pid == getpid (), - "p->pid is expected"); - ok (p->pid_owner == getuid (), - "p->pid_owner is expected"); - diag ("p->cg_path = %s", p->cg_path); - diag ("p->cg_owner = %d", (int) p->cg_owner); - pid_info_destroy (p); - - ok ((p = pid_info_create (-getpid ())) != NULL, - "pid_info_create (-getpid ()) works"); - ok (p->pid == getpid (), - "p->pid is expected"); - ok (p->pid_owner == getuid (), - "p->pid_owner is expected"); - diag ("p->cg_path = %s", p->cg_path); - diag ("p->cg_owner = %d", (int) p->cg_owner); - pid_info_destroy (p); - pid_kill_tests (); - done_testing (); } diff --git a/t/Makefile.am b/t/Makefile.am index 6693203f..36845bd3 100644 --- a/t/Makefile.am +++ b/t/Makefile.am @@ -21,7 +21,6 @@ TESTSCRIPTS = \ t1002-sign-munge.t \ t1003-sign-curve.t \ t2000-imp-exec.t \ - t2001-imp-kill.t \ t2002-imp-run.t \ t2003-imp-exec-pam.t diff --git a/t/t2001-imp-kill.t b/t/t2001-imp-kill.t deleted file mode 100755 index 796e95a3..00000000 --- a/t/t2001-imp-kill.t +++ /dev/null @@ -1,196 +0,0 @@ -#!/bin/sh -# - -test_description='IMP kill basic functionality test - -Basic flux-imp kill failure handling -' - -# Append --logfile option if FLUX_TESTS_LOGFILE is set in environment: -test -n "$FLUX_TESTS_LOGFILE" && set -- "$@" --logfile -. `dirname $0`/sharness.sh - -flux_imp=${SHARNESS_BUILD_DIRECTORY}/src/imp/flux-imp - -sign=${SHARNESS_BUILD_DIRECTORY}/t/src/sign - -fake_imp_input() { - printf '{"J":"%s"}' $(echo $1 | $sign) -} - -echo "# Using ${flux_imp}" - -test_expect_success 'flux-imp kill: returns error when run with no args' ' - test_must_fail $flux_imp kill -' -test_expect_success 'flux-imp kill: returns error when run with 1 arg' ' - test_must_fail $flux_imp kill 15 -' -test_expect_success 'flux-imp kill: returns error with pid=0' ' - test_must_fail $flux_imp kill 15 0 -' -test_expect_success 'flux-imp kill: returns error with invalid signal' ' - test_must_fail $flux_imp kill -15 0 -' -test_expect_success 'flux-imp kill: checks exec.allowed-users' ' - name=wronguser && - cat <<-EOF >${name}.toml && - [exec] - allowed-users = [] - EOF - ( export FLUX_IMP_CONFIG_PATTERN=${name}.toml && - test_must_fail $flux_imp kill 15 1234 >${name}.log 2>&1 - ) && - test_debug "cat ${name}.log" && - grep "kill command not allowed" ${name}.log && unset name -' -test_expect_success SUDO 'flux-imp kill: sudo: user must be in allowed-users' ' - name=wrongusersudo && - cat <<-EOF >${name}.toml && - allow-sudo = true - [exec] - allowed-users = [] - EOF - test_must_fail $SUDO FLUX_IMP_CONFIG_PATTERN=${name}.toml \ - $flux_imp kill 15 1234 >${name}.log 2>&1 && - test_debug "cat ${name}.log" && - grep -i "kill command not allowed" ${name}.log && unset name -' - -test "$chain_lint" = "t" || test_set_prereq NO_CHAIN_LINT - - -test "$(stat -fc %T /sys/fs/cgroup 2>/dev/null)" = "cgroup2fs" \ - -o "$(stat -fc %T /sys/fs/cgroup/unified 2>/dev/null)" = "cgroup2fs" \ - -o "$(stat -fc %T /sys/fs/cgroup/systemd 2>/dev/null)" = "cgroup2fs" \ - -o "$(stat -fc %T /sys/fs/cgroup/systemd 2>/dev/null)" = "cgroupfs" \ - && test_set_prereq SYSTEMD_CGROUP - -test_expect_success NO_CHAIN_LINT,SYSTEMD_CGROUP 'flux-imp kill: works in unpriv mode' ' - name=unpriv && - cat <<-EOF >${name}.toml - allow-sudo = true - [exec] - allowed-users = [ "$(whoami)" ] - EOF - sleep 300 & pid=$! && - FLUX_IMP_CONFIG_PATTERN=${name}.toml \ - $flux_imp kill 15 $pid >${name}.log 2>&1 && - test_debug "cat ${name}.log" && - test_expect_code 143 wait $pid -' -test_expect_success NO_CHAIN_LINT,SYSTEMD_CGROUP,SUDO 'flux-imp kill: works with sudo' ' - name=sudo && - cat <<-EOF >${name}.toml - allow-sudo = true - [exec] - allowed-users = [ "$(whoami)" ] - EOF - sleep 300 & pid=$! && - $SUDO FLUX_IMP_CONFIG_PATTERN=${name}.toml \ - $flux_imp kill 15 $pid >${name}.log 2>&1 && - test_expect_code 143 wait $pid -' -test_expect_success SYSTEMD_CGROUP 'flux-imp kill: fails for nonexistent pid' ' - name=pid-noexist && - cat <<-EOF >${name}.toml && - allow-sudo = true - [exec] - allowed-users = [ "$(whoami)" ] - EOF - for pid in `seq 10000 12000`; do - kill -s 0 ${pid} >/dev/null 2>&1 || break - done && - ( export FLUX_IMP_CONFIG_PATTERN=${name}.toml && - test_must_fail $flux_imp kill 15 $pid >${name}.log 2>&1 - ) && - test_debug "cat ${name}.log" && - grep "No such file or directory" ${name}.log -' -test_expect_success SUDO,SYSTEMD_CGROUP 'flux-imp kill: fails for nonexistent pid under sudo' ' - name=noexist-pid-sudo && - cat <<-EOF >${name}.toml && - allow-sudo = true - [exec] - allowed-users = [ "$(whoami)" ] - EOF - for pid in `seq 10000 12000`; do - kill -s 0 ${pid} >/dev/null 2>&1 || break - done && - test_must_fail $SUDO FLUX_IMP_CONFIG_PATTERN=${name}.toml \ - $flux_imp kill 15 $pid >${name}.log 2>&1 && - test_debug "cat ${name}.log" && - grep "No such file or directory" ${name}.log -' -test_expect_success 'create config for flux-imp exec and signer' ' - cat <<-EOF >sign-none.toml - allow-sudo = true - [sign] - max-ttl = 30 - default-type = "none" - allowed-types = [ "none" ] - [exec] - allowed-users = [ "$(whoami)" ] - allowed-shells = [ "$(pwd)/sleeper.sh" ] - allow-unprivileged-exec = true - EOF -' - -# Need a setuid IMP for the following kill tests to work. This is required -# because on most systemd systems, sudo may migrate children to a root-owned -# cgroup, and `flux-imp kill` uses the ownership of the containing cgroup -# to determine authorization to signal a process -test_expect_success SUDO 'attempt to create setuid copy of flux-imp for testing' ' - sudo cp $flux_imp . && - sudo chmod 4755 ./flux-imp -' - -# In case tests are running on filesystem with nosuid, check that permissions -# of flux-imp are actually setuid, and if so set SUID_ENABLED prereq -# -test -n "$(find ./flux-imp -perm 4755)" && test_set_prereq SUID_ENABLED - -# In order for the following kill tests to work, the owner of the current -# cgroup must be the current user running the test. -# -test "$(FLUX_IMP_CONFIG_PATTERN=sign-none.toml ./flux-imp kill 0 $$ \ - | sed 's/.*cg_owner=//')" = "$(id -u)" && test_set_prereq USER_CGROUP - -waitfile() -{ - count=0 - while ! grep "$2" $1 >/dev/null 2>&1; do - sleep 0.2 - count=$(($count + 1)) - test $count -gt 20 && break - done - grep "$2" $1 >/dev/null -} - -test_expect_success NO_CHAIN_LINT,SUID_ENABLED,USER_CGROUP \ -'flux-imp kill signals children of another flux-imp' ' - cat <<-EOF >sleeper.sh && - #!/bin/sh - echo "\$@" - printf "\$PPID\n" >$(pwd)/sleeper.pid - /bin/sleep "\$@" & - pid=\$! && - trap "echo got SIGTERM; kill \$pid" TERM - wait - echo sleep exited with \$? >&2 - EOF - chmod +x sleeper.sh && - export FLUX_IMP_CONFIG_PATTERN=sign-none.toml && - fake_imp_input foo | \ - ./flux-imp exec $(pwd)/sleeper.sh 30 >sleeper.out 2>&1 & - pid=$! && - waitfile sleeper.pid ".*" && - test_debug "echo calling flux-imp kill $(cat sleeper.pid)" && - FLUX_IMP_CONFIG_PATTERN=$(pwd)/unpriv.toml \ - ./flux-imp kill 15 $(cat sleeper.pid) && - wait $pid && - test_debug "echo waiting for output file" && - grep "got SIGTERM" sleeper.out -' - -test_done