mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-10 05:17:59 +03:00
ef4c325f25
The function is used to automatically feed a buffer into a pipe which can be used by the command to read contents of the buffer. Rather than passing in a pipe, let's create the pipe inside virCommandSetSendBuffer and directly associate the reader end with the command. This way the ownership of both ends of the pipe will end up with the virCommand right away reducing the need of cleanup in callers. The returned value then can be used just to format the appropriate arguments without worrying about cleanup or failure. Signed-off-by: Peter Krempa <pkrempa@redhat.com> Reviewed-by: Ján Tomko <jtomko@redhat.com>
1270 lines
32 KiB
C
1270 lines
32 KiB
C
/*
|
|
* commandtest.c: Test the libCommand API
|
|
*
|
|
* Copyright (C) 2010-2014 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <sys/stat.h>
|
|
#ifndef WIN32
|
|
# include <sys/wait.h>
|
|
#endif
|
|
#include <fcntl.h>
|
|
|
|
#include "testutils.h"
|
|
#include "internal.h"
|
|
#include "viralloc.h"
|
|
#include "vircommand.h"
|
|
#include "virfile.h"
|
|
#include "virpidfile.h"
|
|
#include "virerror.h"
|
|
#include "virthread.h"
|
|
#include "virstring.h"
|
|
#include "virprocess.h"
|
|
#include "virutil.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
#ifdef WIN32
|
|
|
|
int
|
|
main(void)
|
|
{
|
|
return EXIT_AM_SKIP;
|
|
}
|
|
|
|
#else
|
|
|
|
/* Some UNIX lack it in headers & it doesn't hurt to redeclare */
|
|
extern char **environ;
|
|
|
|
static int checkoutput(const char *testname)
|
|
{
|
|
int ret = -1;
|
|
g_autofree char *expectname = NULL;
|
|
g_autofree char *expectlog = NULL;
|
|
g_autofree char *actualname = NULL;
|
|
g_autofree char *actuallog = NULL;
|
|
|
|
expectname = g_strdup_printf("%s/commanddata/%s.log", abs_srcdir, testname);
|
|
actualname = g_strdup_printf("%s/commandhelper.log", abs_builddir);
|
|
|
|
if (virFileReadAll(expectname, 1024*64, &expectlog) < 0) {
|
|
fprintf(stderr, "cannot read %s\n", expectname);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virFileReadAll(actualname, 1024*64, &actuallog) < 0) {
|
|
fprintf(stderr, "cannot read %s\n", actualname);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (STRNEQ(expectlog, actuallog)) {
|
|
virTestDifference(stderr, expectlog, actuallog);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
if (actualname)
|
|
unlink(actualname);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
* No slot for return status must log error.
|
|
*/
|
|
static int test0(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = NULL;
|
|
|
|
cmd = virCommandNew(abs_builddir "/commandhelper-doesnotexist");
|
|
if (virCommandRun(cmd, NULL) == 0)
|
|
return -1;
|
|
|
|
if (virGetLastErrorCode() == VIR_ERR_OK)
|
|
return -1;
|
|
|
|
virResetLastError();
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
* Capturing return status must not log error.
|
|
*/
|
|
static int test1(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = NULL;
|
|
int status;
|
|
|
|
cmd = virCommandNew(abs_builddir "/commandhelper-doesnotexist");
|
|
if (virCommandRun(cmd, &status) < 0)
|
|
return -1;
|
|
if (status != EXIT_ENOENT)
|
|
return -1;
|
|
|
|
virCommandRawStatus(cmd);
|
|
if (virCommandRun(cmd, &status) < 0)
|
|
return -1;
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status) != EXIT_ENOENT)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Run program (twice), no args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test2(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
int ret;
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
if ((ret = checkoutput("test2")) != 0)
|
|
return ret;
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test2");
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* stdin/out/err + two extra FD open
|
|
*/
|
|
static int test3(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
VIR_AUTOCLOSE newfd1 = dup(STDERR_FILENO);
|
|
VIR_AUTOCLOSE newfd2 = dup(STDERR_FILENO);
|
|
int newfd3 = dup(STDERR_FILENO);
|
|
struct stat before, after;
|
|
|
|
if (fstat(newfd3, &before) < 0) {
|
|
perror("fstat");
|
|
return -1;
|
|
}
|
|
virCommandPassFD(cmd, newfd1, 0);
|
|
virCommandPassFD(cmd, newfd3,
|
|
VIR_COMMAND_PASS_FD_CLOSE_PARENT);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
if (fcntl(newfd1, F_GETFL) < 0 ||
|
|
fcntl(newfd2, F_GETFL) < 0) {
|
|
puts("fds 1/2 were not open");
|
|
return -1;
|
|
}
|
|
|
|
/* We expect newfd3 to be closed, but the
|
|
* fd might have already been reused by
|
|
* the event loop. So if it is open, we
|
|
* check if it matches the stat info we
|
|
* got earlier
|
|
*/
|
|
if (fcntl(newfd3, F_GETFL) >= 0 &&
|
|
fstat(newfd3, &after) >= 0) {
|
|
|
|
if (before.st_ino == after.st_ino &&
|
|
before.st_dev == after.st_dev &&
|
|
before.st_mode == after.st_mode) {
|
|
puts("fd 3 should not be open");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return checkoutput("test3");
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, CWD is /
|
|
* Only stdin/out/err open.
|
|
* Daemonized
|
|
*/
|
|
static int test4(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNewArgList(abs_builddir "/commandhelper",
|
|
"--check-daemonize", NULL);
|
|
g_autofree char *pidfile = virPidFileBuildPath(abs_builddir, "commandhelper");
|
|
pid_t pid;
|
|
int ret = -1;
|
|
|
|
if (!pidfile)
|
|
goto cleanup;
|
|
|
|
virCommandSetPidFile(cmd, pidfile);
|
|
virCommandDaemonize(cmd);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virPidFileRead(abs_builddir, "commandhelper", &pid) < 0) {
|
|
printf("cannot read pidfile\n");
|
|
goto cleanup;
|
|
}
|
|
while (kill(pid, 0) != -1)
|
|
g_usleep(100*1000);
|
|
|
|
ret = checkoutput("test4");
|
|
|
|
cleanup:
|
|
if (pidfile)
|
|
unlink(pidfile);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, no args, inherit filtered ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test5(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
|
|
virCommandAddEnvPassCommon(cmd);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test5");
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, no args, inherit filtered ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test6(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
|
|
virCommandAddEnvPass(cmd, "DISPLAY");
|
|
virCommandAddEnvPass(cmd, "DOESNOTEXIST");
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test6");
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, no args, inherit filtered ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test7(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
|
|
virCommandAddEnvPassCommon(cmd);
|
|
virCommandAddEnvPass(cmd, "DISPLAY");
|
|
virCommandAddEnvPass(cmd, "DOESNOTEXIST");
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test7");
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit filtered ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test8(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
|
|
virCommandAddEnvString(cmd, "USER=bogus");
|
|
virCommandAddEnvString(cmd, "LANG=C");
|
|
virCommandAddEnvPair(cmd, "USER", "also bogus");
|
|
virCommandAddEnvPair(cmd, "USER", "test");
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test8");
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, some args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test9(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
const char* const args[] = { "arg1", "arg2", NULL };
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
virCommandAddArg(cmd, "-version");
|
|
virCommandAddArgPair(cmd, "-log", "bar.log");
|
|
virCommandAddArgSet(cmd, args);
|
|
virCommandAddArgBuffer(cmd, &buf);
|
|
virBufferAddLit(&buf, "arg4");
|
|
virCommandAddArgBuffer(cmd, &buf);
|
|
virCommandAddArgList(cmd, "arg5", "arg6", NULL);
|
|
|
|
if (virBufferUse(&buf)) {
|
|
printf("Buffer not transferred\n");
|
|
return -1;
|
|
}
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test9");
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, some args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test10(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
const char *const args[] = {
|
|
"-version", "-log=bar.log", NULL,
|
|
};
|
|
|
|
virCommandAddArgSet(cmd, args);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test10");
|
|
}
|
|
|
|
/*
|
|
* Run program, some args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test11(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
const char *args[] = {
|
|
abs_builddir "/commandhelper",
|
|
"-version", "-log=bar.log", NULL,
|
|
};
|
|
g_autoptr(virCommand) cmd = virCommandNewArgs(args);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test11");
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open. Set stdin data
|
|
*/
|
|
static int test12(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
|
|
virCommandSetInputBuffer(cmd, "Hello World\n");
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test12");
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open. Set stdin data
|
|
*/
|
|
static int test13(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
virCommandPtr cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
g_autofree char *outactual = NULL;
|
|
const char *outexpect = "BEGIN STDOUT\n"
|
|
"Hello World\n"
|
|
"END STDOUT\n";
|
|
int ret = -1;
|
|
|
|
virCommandSetInputBuffer(cmd, "Hello World\n");
|
|
virCommandSetOutputBuffer(cmd, &outactual);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
if (!outactual)
|
|
goto cleanup;
|
|
|
|
virCommandFree(cmd);
|
|
cmd = NULL;
|
|
|
|
if (STRNEQ(outactual, outexpect)) {
|
|
virTestDifference(stderr, outexpect, outactual);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = checkoutput("test13");
|
|
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* Only stdin/out/err open. Set stdin data
|
|
*/
|
|
static int test14(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
virCommandPtr cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
g_autofree char *outactual = NULL;
|
|
const char *outexpect = "BEGIN STDOUT\n"
|
|
"Hello World\n"
|
|
"END STDOUT\n";
|
|
g_autofree char *erractual = NULL;
|
|
const char *errexpect = "BEGIN STDERR\n"
|
|
"Hello World\n"
|
|
"END STDERR\n";
|
|
|
|
g_autofree char *jointactual = NULL;
|
|
const char *jointexpect = "BEGIN STDOUT\n"
|
|
"BEGIN STDERR\n"
|
|
"Hello World\n"
|
|
"Hello World\n"
|
|
"END STDOUT\n"
|
|
"END STDERR\n";
|
|
int ret = -1;
|
|
|
|
virCommandSetInputBuffer(cmd, "Hello World\n");
|
|
virCommandSetOutputBuffer(cmd, &outactual);
|
|
virCommandSetErrorBuffer(cmd, &erractual);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
if (!outactual || !erractual)
|
|
goto cleanup;
|
|
|
|
virCommandFree(cmd);
|
|
|
|
cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
virCommandSetInputBuffer(cmd, "Hello World\n");
|
|
virCommandSetOutputBuffer(cmd, &jointactual);
|
|
virCommandSetErrorBuffer(cmd, &jointactual);
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
if (!jointactual)
|
|
goto cleanup;
|
|
|
|
if (STRNEQ(outactual, outexpect)) {
|
|
virTestDifference(stderr, outexpect, outactual);
|
|
goto cleanup;
|
|
}
|
|
if (STRNEQ(erractual, errexpect)) {
|
|
virTestDifference(stderr, errexpect, erractual);
|
|
goto cleanup;
|
|
}
|
|
if (STRNEQ(jointactual, jointexpect)) {
|
|
virTestDifference(stderr, jointexpect, jointactual);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = checkoutput("test14");
|
|
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, change CWD.
|
|
* Only stdin/out/err open
|
|
*/
|
|
static int test15(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
g_autofree char *cwd = NULL;
|
|
|
|
cwd = g_strdup_printf("%s/commanddata", abs_srcdir);
|
|
virCommandSetWorkingDirectory(cmd, cwd);
|
|
virCommandSetUmask(cmd, 002);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test15");
|
|
}
|
|
|
|
/*
|
|
* Don't run program; rather, log what would be run.
|
|
*/
|
|
static int test16(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew("true");
|
|
g_autofree char *outactual = NULL;
|
|
const char *outexpect = "A=B C='D E' true F 'G H'";
|
|
VIR_AUTOCLOSE fd = -1;
|
|
|
|
virCommandAddEnvPair(cmd, "A", "B");
|
|
virCommandAddEnvPair(cmd, "C", "D E");
|
|
virCommandAddArg(cmd, "F");
|
|
virCommandAddArg(cmd, "G H");
|
|
|
|
if ((outactual = virCommandToString(cmd, false)) == NULL) {
|
|
printf("Cannot convert to string: %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
if ((fd = open(abs_builddir "/commandhelper.log",
|
|
O_CREAT | O_TRUNC | O_WRONLY, 0600)) < 0) {
|
|
printf("Cannot open log file: %s\n", g_strerror(errno));
|
|
return -1;
|
|
}
|
|
virCommandWriteArgLog(cmd, fd);
|
|
if (VIR_CLOSE(fd) < 0) {
|
|
printf("Cannot close log file: %s\n", g_strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
if (STRNEQ(outactual, outexpect)) {
|
|
virTestDifference(stderr, outexpect, outactual);
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test16");
|
|
}
|
|
|
|
/*
|
|
* Test string handling when no output is present.
|
|
*/
|
|
static int test17(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew("true");
|
|
int ret = -1;
|
|
char *outbuf = NULL;
|
|
g_autofree char *errbuf = NULL;
|
|
|
|
virCommandSetOutputBuffer(cmd, &outbuf);
|
|
if (outbuf != NULL) {
|
|
puts("buffer not sanitized at registration");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
|
|
sa_assert(outbuf);
|
|
if (*outbuf) {
|
|
puts("output buffer is not an allocated empty string");
|
|
goto cleanup;
|
|
}
|
|
VIR_FREE(outbuf);
|
|
outbuf = g_strdup("should not be leaked");
|
|
|
|
virCommandSetErrorBuffer(cmd, &errbuf);
|
|
if (errbuf != NULL) {
|
|
puts("buffer not sanitized at registration");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
|
|
if (*outbuf || *errbuf) {
|
|
puts("output buffers are not allocated empty strings");
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FREE(outbuf);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Run long-running daemon, to ensure no hang.
|
|
*/
|
|
static int test18(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
virCommandPtr cmd = virCommandNewArgList("sleep", "100", NULL);
|
|
g_autofree char *pidfile = virPidFileBuildPath(abs_builddir, "commandhelper");
|
|
pid_t pid;
|
|
int ret = -1;
|
|
|
|
if (!pidfile)
|
|
goto cleanup;
|
|
|
|
virCommandSetPidFile(cmd, pidfile);
|
|
virCommandDaemonize(cmd);
|
|
|
|
alarm(5);
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
alarm(0);
|
|
|
|
if (virPidFileRead(abs_builddir, "commandhelper", &pid) < 0) {
|
|
printf("cannot read pidfile\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
virCommandFree(cmd);
|
|
cmd = NULL;
|
|
if (kill(pid, 0) != 0) {
|
|
printf("daemon should still be running\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
while (kill(pid, SIGINT) != -1)
|
|
g_usleep(100*1000);
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
if (pidfile)
|
|
unlink(pidfile);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Asynchronously run long-running daemon, to ensure no hang.
|
|
*/
|
|
static int test19(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNewArgList("sleep", "100", NULL);
|
|
pid_t pid;
|
|
|
|
alarm(5);
|
|
if (virCommandRunAsync(cmd, &pid) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
if (kill(pid, 0) != 0) {
|
|
printf("Child should still be running");
|
|
return -1;
|
|
}
|
|
|
|
virCommandAbort(cmd);
|
|
|
|
if (kill(pid, 0) == 0) {
|
|
printf("Child should be aborted");
|
|
return -1;
|
|
}
|
|
|
|
alarm(0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Run program, no args, inherit all ENV, keep CWD.
|
|
* Ignore huge stdin data, to provoke SIGPIPE or EPIPE in parent.
|
|
*/
|
|
static int test20(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNewArgList(abs_builddir "/commandhelper",
|
|
"--close-stdin", NULL);
|
|
g_autofree char *buf = NULL;
|
|
|
|
struct sigaction sig_action;
|
|
|
|
sig_action.sa_handler = SIG_IGN;
|
|
sig_action.sa_flags = 0;
|
|
sigemptyset(&sig_action.sa_mask);
|
|
|
|
sigaction(SIGPIPE, &sig_action, NULL);
|
|
|
|
buf = g_strdup_printf("1\n%100000d\n", 2);
|
|
virCommandSetInputBuffer(cmd, buf);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test20");
|
|
}
|
|
|
|
static const char *const newenv[] = {
|
|
"PATH=/usr/bin:/bin",
|
|
"HOSTNAME=test",
|
|
"LANG=C",
|
|
"HOME=/home/test",
|
|
"USER=test",
|
|
"LOGNAME=test",
|
|
"TMPDIR=/tmp",
|
|
"DISPLAY=:0.0",
|
|
NULL
|
|
};
|
|
|
|
static int test21(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
const char *wrbuf = "Hello world\n";
|
|
g_autofree char *outbuf = NULL;
|
|
g_autofree char *errbuf = NULL;
|
|
const char *outbufExpected = "BEGIN STDOUT\n"
|
|
"Hello world\n"
|
|
"END STDOUT\n";
|
|
const char *errbufExpected = "BEGIN STDERR\n"
|
|
"Hello world\n"
|
|
"END STDERR\n";
|
|
|
|
virCommandSetInputBuffer(cmd, wrbuf);
|
|
virCommandSetOutputBuffer(cmd, &outbuf);
|
|
virCommandSetErrorBuffer(cmd, &errbuf);
|
|
virCommandDoAsyncIO(cmd);
|
|
|
|
if (virCommandRunAsync(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
|
|
if (virCommandWait(cmd, NULL) < 0)
|
|
return -1;
|
|
|
|
if (virTestGetVerbose())
|
|
printf("STDOUT:%s\nSTDERR:%s\n", NULLSTR(outbuf), NULLSTR(errbuf));
|
|
|
|
if (STRNEQ_NULLABLE(outbuf, outbufExpected)) {
|
|
virTestDifference(stderr, outbufExpected, outbuf);
|
|
return -1;
|
|
}
|
|
|
|
if (STRNEQ_NULLABLE(errbuf, errbufExpected)) {
|
|
virTestDifference(stderr, errbufExpected, errbuf);
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test21");
|
|
}
|
|
|
|
static int
|
|
test22(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
int ret = -1;
|
|
virCommandPtr cmd;
|
|
int status = -1;
|
|
|
|
cmd = virCommandNewArgList("/bin/sh", "-c", "exit 3", NULL);
|
|
|
|
if (virCommandRun(cmd, &status) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
if (status != 3) {
|
|
printf("Unexpected status %d\n", status);
|
|
goto cleanup;
|
|
}
|
|
|
|
virCommandRawStatus(cmd);
|
|
if (virCommandRun(cmd, &status) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status) != 3) {
|
|
printf("Unexpected status %d\n", status);
|
|
goto cleanup;
|
|
}
|
|
|
|
virCommandFree(cmd);
|
|
cmd = virCommandNewArgList("/bin/sh", "-c", "kill -9 $$", NULL);
|
|
|
|
if (virCommandRun(cmd, &status) == 0) {
|
|
printf("Death by signal not detected, status %d\n", status);
|
|
goto cleanup;
|
|
}
|
|
|
|
virCommandRawStatus(cmd);
|
|
if (virCommandRun(cmd, &status) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
if (!WIFSIGNALED(status) || WTERMSIG(status) != SIGKILL) {
|
|
printf("Unexpected status %d\n", status);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int
|
|
test23(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
/* Not strictly a virCommand test, but this is the easiest place
|
|
* to test this lower-level interface. It takes a double fork to
|
|
* test virProcessExitWithStatus. */
|
|
int status = -1;
|
|
pid_t pid;
|
|
|
|
if ((pid = virFork()) < 0)
|
|
return -1;
|
|
if (pid == 0) {
|
|
if ((pid = virFork()) < 0)
|
|
_exit(EXIT_FAILURE);
|
|
if (pid == 0)
|
|
_exit(42);
|
|
if (virProcessWait(pid, &status, true) < 0)
|
|
_exit(EXIT_FAILURE);
|
|
virProcessExitWithStatus(status);
|
|
_exit(EXIT_FAILURE);
|
|
}
|
|
|
|
if (virProcessWait(pid, &status, true) < 0)
|
|
return -1;
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status) != 42) {
|
|
printf("Unexpected status %d\n", status);
|
|
return -1;
|
|
}
|
|
|
|
if ((pid = virFork()) < 0)
|
|
return -1;
|
|
if (pid == 0) {
|
|
if ((pid = virFork()) < 0)
|
|
_exit(EXIT_FAILURE);
|
|
if (pid == 0) {
|
|
raise(SIGKILL);
|
|
_exit(EXIT_FAILURE);
|
|
}
|
|
if (virProcessWait(pid, &status, true) < 0)
|
|
_exit(EXIT_FAILURE);
|
|
virProcessExitWithStatus(status);
|
|
_exit(EXIT_FAILURE);
|
|
}
|
|
|
|
if (virProcessWait(pid, &status, true) < 0)
|
|
return -1;
|
|
if (!WIFSIGNALED(status) || WTERMSIG(status) != SIGKILL) {
|
|
printf("Unexpected status %d\n", status);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int test25(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
int ret = -1;
|
|
int pipeFD[2] = { -1, -1};
|
|
int rv = 0;
|
|
ssize_t tries = 100;
|
|
pid_t pid;
|
|
g_autofree gid_t *groups = NULL;
|
|
int ngroups;
|
|
g_autoptr(virCommand) cmd = virCommandNew("some/nonexistent/binary");
|
|
|
|
if (virPipeQuiet(pipeFD) < 0) {
|
|
fprintf(stderr, "Unable to create pipe\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virSetNonBlock(pipeFD[0]) < 0) {
|
|
fprintf(stderr, "Unable to make read end of pipe nonblocking\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if ((ngroups = virGetGroupList(virCommandGetUID(cmd), virCommandGetGID(cmd),
|
|
&groups)) < 0)
|
|
goto cleanup;
|
|
|
|
/* Now, fork and try to exec a nonexistent binary. */
|
|
pid = virFork();
|
|
if (pid < 0) {
|
|
fprintf(stderr, "Unable to spawn child\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (pid == 0) {
|
|
/* Child */
|
|
rv = virCommandExec(cmd, groups, ngroups);
|
|
|
|
if (safewrite(pipeFD[1], &rv, sizeof(rv)) < 0)
|
|
fprintf(stderr, "Unable to write to pipe\n");
|
|
_exit(EXIT_FAILURE);
|
|
}
|
|
|
|
/* Parent */
|
|
while (--tries) {
|
|
if (saferead(pipeFD[0], &rv, sizeof(rv)) < 0) {
|
|
if (errno != EWOULDBLOCK) {
|
|
fprintf(stderr, "Unable to read from pipe\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
g_usleep(10 * 1000);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!tries) {
|
|
fprintf(stderr, "Child hasn't returned anything\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (rv >= 0) {
|
|
fprintf(stderr, "Child should have returned an error\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FORCE_CLOSE(pipeFD[0]);
|
|
VIR_FORCE_CLOSE(pipeFD[1]);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Don't run program; rather, log what would be run.
|
|
*/
|
|
static int test26(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew("true");
|
|
g_autofree char *outactual = NULL;
|
|
const char *outexpect =
|
|
"A=B \\\n"
|
|
"C='D E' \\\n"
|
|
"true \\\n"
|
|
"--foo bar \\\n"
|
|
"--oooh \\\n"
|
|
"-f \\\n"
|
|
"--wizz 'eek eek' \\\n"
|
|
"-w \\\n"
|
|
"-z \\\n"
|
|
"-l \\\n"
|
|
"--mmm flash \\\n"
|
|
"bang \\\n"
|
|
"wallop";
|
|
|
|
VIR_AUTOCLOSE fd = -1;
|
|
|
|
virCommandAddEnvPair(cmd, "A", "B");
|
|
virCommandAddEnvPair(cmd, "C", "D E");
|
|
virCommandAddArgList(cmd, "--foo", "bar", "--oooh", "-f",
|
|
"--wizz", "eek eek", "-w", "-z", "-l",
|
|
"--mmm", "flash", "bang", "wallop",
|
|
NULL);
|
|
|
|
if ((outactual = virCommandToString(cmd, true)) == NULL) {
|
|
printf("Cannot convert to string: %s\n", virGetLastErrorMessage());
|
|
return -1;
|
|
}
|
|
if ((fd = open(abs_builddir "/commandhelper.log",
|
|
O_CREAT | O_TRUNC | O_WRONLY, 0600)) < 0) {
|
|
printf("Cannot open log file: %s\n", g_strerror(errno));
|
|
return -1;
|
|
}
|
|
virCommandWriteArgLog(cmd, fd);
|
|
if (VIR_CLOSE(fd) < 0) {
|
|
printf("Cannot close log file: %s\n", g_strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
if (STRNEQ(outactual, outexpect)) {
|
|
virTestDifference(stderr, outexpect, outactual);
|
|
return -1;
|
|
}
|
|
|
|
return checkoutput("test26");
|
|
}
|
|
|
|
static int test27(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
g_autoptr(virCommand) cmd = virCommandNew(abs_builddir "/commandhelper");
|
|
int buf1fd;
|
|
int buf2fd;
|
|
int ret = -1;
|
|
size_t buflen = 1024 * 128;
|
|
g_autofree char *buffer0 = NULL;
|
|
g_autofree char *buffer1 = NULL;
|
|
g_autofree char *buffer2 = NULL;
|
|
g_autofree char *outactual = NULL;
|
|
g_autofree char *erractual = NULL;
|
|
g_autofree char *outexpect = NULL;
|
|
# define TEST27_OUTEXPECT_TEMP "BEGIN STDOUT\n" \
|
|
"%s%s%s" \
|
|
"END STDOUT\n"
|
|
g_autofree char *errexpect = NULL;
|
|
# define TEST27_ERREXPECT_TEMP "BEGIN STDERR\n" \
|
|
"%s%s%s" \
|
|
"END STDERR\n"
|
|
|
|
buffer0 = g_new0(char, buflen);
|
|
buffer1 = g_new0(char, buflen);
|
|
buffer2 = g_new0(char, buflen);
|
|
|
|
memset(buffer0, 'H', buflen - 2);
|
|
buffer0[buflen - 2] = '\n';
|
|
buffer0[buflen - 1] = 0;
|
|
|
|
memset(buffer1, '1', buflen - 2);
|
|
buffer1[buflen - 2] = '\n';
|
|
buffer1[buflen - 1] = 0;
|
|
|
|
memset(buffer2, '2', buflen - 2);
|
|
buffer2[buflen - 2] = '\n';
|
|
buffer2[buflen - 1] = 0;
|
|
|
|
outexpect = g_strdup_printf(TEST27_OUTEXPECT_TEMP,
|
|
buffer0, buffer1, buffer2);
|
|
errexpect = g_strdup_printf(TEST27_ERREXPECT_TEMP,
|
|
buffer0, buffer1, buffer2);
|
|
|
|
buf1fd = virCommandSetSendBuffer(cmd, (unsigned char *) g_steal_pointer(&buffer1), buflen - 1);
|
|
buf2fd = virCommandSetSendBuffer(cmd, (unsigned char *) g_steal_pointer(&buffer2), buflen - 1);
|
|
|
|
virCommandAddArg(cmd, "--readfd");
|
|
virCommandAddArgFormat(cmd, "%d", buf1fd);
|
|
|
|
virCommandAddArg(cmd, "--readfd");
|
|
virCommandAddArgFormat(cmd, "%d", buf2fd);
|
|
|
|
virCommandSetInputBuffer(cmd, buffer0);
|
|
virCommandSetOutputBuffer(cmd, &outactual);
|
|
virCommandSetErrorBuffer(cmd, &erractual);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0) {
|
|
printf("Cannot run child %s\n", virGetLastErrorMessage());
|
|
goto cleanup;
|
|
}
|
|
|
|
if (!outactual || !erractual)
|
|
goto cleanup;
|
|
|
|
if (STRNEQ(outactual, outexpect)) {
|
|
virTestDifference(stderr, outexpect, outactual);
|
|
goto cleanup;
|
|
}
|
|
if (STRNEQ(erractual, errexpect)) {
|
|
virTestDifference(stderr, errexpect, erractual);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (checkoutput("test27") < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int
|
|
test28Callback(pid_t pid G_GNUC_UNUSED,
|
|
void *opaque G_GNUC_UNUSED)
|
|
{
|
|
virReportSystemError(ENODATA, "%s", "some error message");
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int
|
|
test28(const void *unused G_GNUC_UNUSED)
|
|
{
|
|
/* Not strictly a virCommand test, but this is the easiest place
|
|
* to test this lower-level interface. */
|
|
virErrorPtr err;
|
|
g_autofree char *msg = g_strdup_printf("some error message: %s", g_strerror(ENODATA));
|
|
|
|
if (virProcessRunInFork(test28Callback, NULL) != -1) {
|
|
fprintf(stderr, "virProcessRunInFork did not fail\n");
|
|
return -1;
|
|
}
|
|
|
|
if (!(err = virGetLastError())) {
|
|
fprintf(stderr, "Expected error but got nothing\n");
|
|
return -1;
|
|
}
|
|
|
|
if (!(err->code == VIR_ERR_SYSTEM_ERROR &&
|
|
err->domain == 0 &&
|
|
STREQ(err->message, msg) &&
|
|
err->level == VIR_ERR_ERROR &&
|
|
STREQ(err->str1, "%s") &&
|
|
STREQ(err->str2, msg) &&
|
|
err->int1 == ENODATA &&
|
|
err->int2 == -1)) {
|
|
fprintf(stderr, "Unexpected error object\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int
|
|
mymain(void)
|
|
{
|
|
int ret = 0;
|
|
int fd;
|
|
int virinitret;
|
|
|
|
if (chdir("/tmp") < 0)
|
|
return EXIT_FAILURE;
|
|
|
|
umask(022);
|
|
|
|
setpgid(0, 0);
|
|
ignore_value(setsid());
|
|
|
|
/* Our test expects particular fd values; to get that, we must not
|
|
* leak fds that we inherited from a lazy parent. At the same
|
|
* time, virInitialize may open some fds (perhaps via third-party
|
|
* libraries that it uses), and we must not kill off an fd that
|
|
* this process opens as it might break expectations of a
|
|
* pthread_atfork handler, as well as interfering with our tests
|
|
* trying to ensure we aren't leaking to our children. The
|
|
* solution is to do things in two phases - reserve the fds we
|
|
* want by overwriting any externally inherited fds, then
|
|
* initialize, then clear the slots for testing. */
|
|
if ((fd = open("/dev/null", O_RDONLY)) < 0 ||
|
|
dup2(fd, 3) < 0 ||
|
|
dup2(fd, 4) < 0 ||
|
|
dup2(fd, 5) < 0 ||
|
|
dup2(fd, 6) < 0 ||
|
|
dup2(fd, 7) < 0 ||
|
|
dup2(fd, 8) < 0 ||
|
|
(fd > 8 && VIR_CLOSE(fd) < 0)) {
|
|
VIR_FORCE_CLOSE(fd);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* Prime the debug/verbose settings from the env vars,
|
|
* since we're about to reset 'environ' */
|
|
ignore_value(virTestGetDebug());
|
|
ignore_value(virTestGetVerbose());
|
|
|
|
/* Make sure to not leak fd's */
|
|
virinitret = virInitialize();
|
|
|
|
/* Phase two of killing interfering fds; see above. */
|
|
/* coverity[overwrite_var] - silence the obvious */
|
|
fd = 3;
|
|
VIR_FORCE_CLOSE(fd);
|
|
fd = 4;
|
|
VIR_FORCE_CLOSE(fd);
|
|
fd = 5;
|
|
VIR_FORCE_CLOSE(fd);
|
|
fd = 6;
|
|
VIR_FORCE_CLOSE(fd);
|
|
fd = 7;
|
|
VIR_FORCE_CLOSE(fd);
|
|
fd = 8;
|
|
VIR_FORCE_CLOSE(fd);
|
|
|
|
if (virinitret < 0)
|
|
return EXIT_FAILURE;
|
|
|
|
environ = (char **)newenv;
|
|
|
|
# define DO_TEST(NAME) \
|
|
if (virTestRun("Command Exec " #NAME " test", \
|
|
NAME, NULL) < 0) \
|
|
ret = -1
|
|
|
|
DO_TEST(test0);
|
|
DO_TEST(test1);
|
|
DO_TEST(test2);
|
|
DO_TEST(test3);
|
|
DO_TEST(test4);
|
|
DO_TEST(test5);
|
|
DO_TEST(test6);
|
|
DO_TEST(test7);
|
|
DO_TEST(test8);
|
|
DO_TEST(test9);
|
|
DO_TEST(test10);
|
|
DO_TEST(test11);
|
|
DO_TEST(test12);
|
|
DO_TEST(test13);
|
|
DO_TEST(test14);
|
|
DO_TEST(test15);
|
|
DO_TEST(test16);
|
|
DO_TEST(test17);
|
|
DO_TEST(test18);
|
|
DO_TEST(test19);
|
|
DO_TEST(test20);
|
|
DO_TEST(test21);
|
|
DO_TEST(test22);
|
|
DO_TEST(test23);
|
|
DO_TEST(test25);
|
|
DO_TEST(test26);
|
|
DO_TEST(test27);
|
|
DO_TEST(test28);
|
|
|
|
return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
|
|
}
|
|
|
|
VIR_TEST_MAIN(mymain)
|
|
|
|
#endif /* !WIN32 */
|