From 6d6e91cd2d6c567e772536fd23babafd00d0940d Mon Sep 17 00:00:00 2001 From: Gunnar Beutner Date: Mon, 21 Apr 2014 14:39:35 +0200 Subject: [PATCH] Implement Windows support for the Process class. Fixes #3684 --- lib/base/CMakeLists.txt | 5 +- lib/base/process-unix.cpp | 360 ------------------ lib/base/process-windows.cpp | 34 -- lib/base/process.cpp | 495 ++++++++++++++++++++++++- lib/base/process.h | 25 +- lib/methods/pluginchecktask.cpp | 2 +- lib/methods/plugineventtask.cpp | 2 +- lib/methods/pluginnotificationtask.cpp | 2 +- 8 files changed, 511 insertions(+), 414 deletions(-) delete mode 100644 lib/base/process-unix.cpp delete mode 100644 lib/base/process-windows.cpp diff --git a/lib/base/CMakeLists.txt b/lib/base/CMakeLists.txt index 7709d2a0a..c14924b56 100644 --- a/lib/base/CMakeLists.txt +++ b/lib/base/CMakeLists.txt @@ -27,9 +27,8 @@ add_library(base SHARED convert.cpp dictionary.cpp dynamicobject.cpp dynamicobject.th dynamictype.cpp exception.cpp fifo.cpp filelogger.cpp filelogger.th logger.cpp logger.th netstring.cpp networkstream.cpp object.cpp objectlock.cpp process.cpp - process-unix.cpp process-windows.cpp qstring.cpp ringbuffer.cpp - scriptfunction.cpp scriptfunctionwrapper.cpp scriptutils.cpp - scriptvariable.cpp serializer.cpp socket.cpp stacktrace.cpp + qstring.cpp ringbuffer.cpp scriptfunction.cpp scriptfunctionwrapper.cpp + scriptutils.cpp scriptvariable.cpp serializer.cpp socket.cpp stacktrace.cpp statsfunction.cpp stdiostream.cpp stream_bio.cpp stream.cpp streamlogger.cpp streamlogger.th sysloglogger.cpp sysloglogger.th tcpsocket.cpp threadpool.cpp timer.cpp tlsstream.cpp tlsutility.cpp type.cpp unixsocket.cpp utility.cpp value.cpp diff --git a/lib/base/process-unix.cpp b/lib/base/process-unix.cpp deleted file mode 100644 index fba0068b7..000000000 --- a/lib/base/process-unix.cpp +++ /dev/null @@ -1,360 +0,0 @@ -/****************************************************************************** - * Icinga 2 * - * Copyright (C) 2012-2014 Icinga Development Team (http://www.icinga.org) * - * * - * 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 Free Software Foundation; either version 2 * - * of the License, or (at your option) any later version. * - * * - * This program 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 General Public License for more details. * - * * - * You should have received a copy of the GNU General Public License * - * along with this program; if not, write to the Free Software Foundation * - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. * - ******************************************************************************/ - -#include "base/process.h" -#include "base/exception.h" -#include "base/convert.h" -#include "base/objectlock.h" -#include "base/initialize.h" -#include "base/logger_fwd.h" -#include "base/utility.h" -#include "base/scriptvariable.h" -#include -#include -#include -#include -#include - -#ifndef _WIN32 -#include -#include - -using namespace icinga; - -#ifndef __APPLE__ -extern char **environ; -#else /* __APPLE__ */ -#include -#define environ (*_NSGetEnviron()) -#endif /* __APPLE__ */ - -#define IOTHREADS 8 - -static boost::mutex l_ProcessMutex[IOTHREADS]; -static std::map l_Processes[IOTHREADS]; -static int l_EventFDs[IOTHREADS][2]; -static boost::once_flag l_OnceFlag = BOOST_ONCE_INIT; - -INITIALIZE_ONCE(&Process::StaticInitialize); - -void Process::StaticInitialize(void) -{ - for (int tid = 0; tid < IOTHREADS; tid++) { - #ifdef HAVE_PIPE2 - if (pipe2(l_EventFDs[tid], O_CLOEXEC) < 0) { - BOOST_THROW_EXCEPTION(posix_error() - << boost::errinfo_api_function("pipe2") - << boost::errinfo_errno(errno)); - } -#else /* HAVE_PIPE2 */ - if (pipe(l_EventFDs[tid]) < 0) { - BOOST_THROW_EXCEPTION(posix_error() - << boost::errinfo_api_function("pipe") - << boost::errinfo_errno(errno)); - } - - Utility::SetCloExec(l_EventFDs[tid][0]); - Utility::SetCloExec(l_EventFDs[tid][1]); -#endif /* HAVE_PIPE2 */ - - Utility::SetNonBlocking(l_EventFDs[tid][0]); - Utility::SetNonBlocking(l_EventFDs[tid][1]); - } -} - -void Process::ThreadInitialize(void) -{ - /* Note to self: Make sure this runs _after_ we've daemonized. */ - for (int tid = 0; tid < IOTHREADS; tid++) { - boost::thread t(boost::bind(&Process::IOThreadProc, tid)); - t.detach(); - } -} - -void Process::IOThreadProc(int tid) -{ - pollfd *pfds = NULL; - int count = 0; - - Utility::SetThreadName("ProcessIO"); - - for (;;) { - double now, timeout = -1; - - now = Utility::GetTime(); - - { - boost::mutex::scoped_lock lock(l_ProcessMutex[tid]); - - count = 1 + l_Processes[tid].size(); - pfds = reinterpret_cast(realloc(pfds, sizeof(pollfd) * count)); - - pfds[0].fd = l_EventFDs[tid][0]; - pfds[0].events = POLLIN; - pfds[0].revents = 0; - - int i = 1; - std::pair kv; - BOOST_FOREACH(kv, l_Processes[tid]) { - pfds[i].fd = kv.second->m_FD; - pfds[i].events = POLLIN; - pfds[i].revents = 0; - - if (kv.second->m_Timeout != 0) { - double delta = kv.second->m_Timeout - (now - kv.second->m_Result.ExecutionStart); - - if (timeout == -1 || delta < timeout) - timeout = delta; - } - - i++; - } - } - - if (timeout != -1) - timeout *= 1000; - - int rc = poll(pfds, count, timeout); - - if (rc < 0) - continue; - - { - boost::mutex::scoped_lock lock(l_ProcessMutex[tid]); - - if (pfds[0].revents & (POLLIN|POLLHUP|POLLERR)) { - char buffer[512]; - (void) read(l_EventFDs[tid][0], buffer, sizeof(buffer)); - } - - for (int i = 1; i < count; i++) { - if (pfds[i].revents & (POLLIN|POLLHUP|POLLERR)) { - std::map::iterator it; - it = l_Processes[tid].find(pfds[i].fd); - - if (it == l_Processes[tid].end()) - continue; /* This should never happen. */ - - if (!it->second->DoEvents()) { - (void) close(it->first); - l_Processes[tid].erase(it); - } - } - } - } - } -} - -void Process::Run(const boost::function& callback) -{ - boost::call_once(l_OnceFlag, &Process::ThreadInitialize); - - m_Result.ExecutionStart = Utility::GetTime(); - - int fds[2]; - -#ifdef HAVE_PIPE2 - if (pipe2(fds, O_CLOEXEC) < 0) { - BOOST_THROW_EXCEPTION(posix_error() - << boost::errinfo_api_function("pipe2") - << boost::errinfo_errno(errno)); - } -#else /* HAVE_PIPE2 */ - if (pipe(fds) < 0) { - BOOST_THROW_EXCEPTION(posix_error() - << boost::errinfo_api_function("pipe") - << boost::errinfo_errno(errno)); - } - - Utility::SetCloExec(fds[0]); - Utility::SetCloExec(fds[1]); -#endif /* HAVE_PIPE2 */ - - // build argv - char **argv = new char *[m_Arguments.size() + 1]; - - for (unsigned int i = 0; i < m_Arguments.size(); i++) - argv[i] = strdup(m_Arguments[i].CStr()); - - argv[m_Arguments.size()] = NULL; - - // build envp - int envc = 0; - - /* count existing environment variables */ - while (environ[envc] != NULL) - envc++; - - char **envp = new char *[envc + (m_ExtraEnvironment ? m_ExtraEnvironment->GetLength() : 0) + 1]; - - for (int i = 0; i < envc; i++) - envp[i] = strdup(environ[i]); - - if (m_ExtraEnvironment) { - ObjectLock olock(m_ExtraEnvironment); - - int index = envc; - BOOST_FOREACH(const Dictionary::Pair& kv, m_ExtraEnvironment) { - String skv = kv.first + "=" + Convert::ToString(kv.second); - envp[index] = strdup(skv.CStr()); - index++; - } - } - - envp[envc + (m_ExtraEnvironment ? m_ExtraEnvironment->GetLength() : 0)] = NULL; - - m_ExtraEnvironment.reset(); - -#ifdef HAVE_VFORK - Value use_vfork = ScriptVariable::Get("UseVfork"); - - if (use_vfork.IsEmpty() || static_cast(use_vfork)) - m_Pid = vfork(); - else - m_Pid = fork(); -#else /* HAVE_VFORK */ - m_Pid = fork(); -#endif /* HAVE_VFORK */ - - if (m_Pid < 0) { - BOOST_THROW_EXCEPTION(posix_error() - << boost::errinfo_api_function("fork") - << boost::errinfo_errno(errno)); - } - - if (m_Pid == 0) { - // child process - - if (dup2(fds[1], STDOUT_FILENO) < 0 || dup2(fds[1], STDERR_FILENO) < 0) { - perror("dup2() failed."); - _exit(128); - } - - (void) close(fds[0]); - (void) close(fds[1]); - - (void) nice(5); - - if (icinga2_execvpe(argv[0], argv, envp) < 0) { - perror("execvpe() failed."); - _exit(128); - } - - _exit(128); - } - - // parent process - - Log(LogDebug, "base", "Running command '" + boost::algorithm::join(m_Arguments, " ") + - "': PID " + Convert::ToString(m_Pid)); - - m_Arguments.clear(); - - // free arguments - for (int i = 0; argv[i] != NULL; i++) - free(argv[i]); - - delete [] argv; - - // free environment - for (int i = 0; envp[i] != NULL; i++) - free(envp[i]); - - delete [] envp; - - (void) close(fds[1]); - - Utility::SetNonBlocking(fds[0]); - - m_FD = fds[0]; - m_Callback = callback; - - int tid = GetTID(); - - { - boost::mutex::scoped_lock lock(l_ProcessMutex[tid]); - l_Processes[tid][m_FD] = GetSelf(); - } - - (void) write(l_EventFDs[tid][1], "T", 1); -} - -bool Process::DoEvents(void) -{ - if (m_Timeout != 0) { - double timeout = m_Timeout - (Utility::GetTime() - m_Result.ExecutionStart); - - if (timeout < 0) { - m_OutputStream << ""; - kill(m_Pid, SIGKILL); - } - } - - char buffer[512]; - for (;;) { - int rc = read(m_FD, buffer, sizeof(buffer)); - - if (rc < 0 && (errno == EAGAIN || errno == EWOULDBLOCK)) - return true; - - if (rc > 0) { - m_OutputStream.write(buffer, rc); - continue; - } - - break; - } - - String output = m_OutputStream.str(); - - int status, exitcode; - - if (waitpid(m_Pid, &status, 0) != m_Pid) { - BOOST_THROW_EXCEPTION(posix_error() - << boost::errinfo_api_function("waitpid") - << boost::errinfo_errno(errno)); - } - - if (WIFEXITED(status)) { - exitcode = WEXITSTATUS(status); - } else if (WIFSIGNALED(status)) { - std::ostringstream outputbuf; - outputbuf << ""; - output = output + outputbuf.str(); - exitcode = 128; - } else { - exitcode = 128; - } - - m_Result.ExecutionEnd = Utility::GetTime(); - m_Result.ExitStatus = exitcode; - m_Result.Output = output; - - if (m_Callback) - Utility::QueueAsyncCallback(boost::bind(m_Callback, m_Result)); - - return false; -} - -int Process::GetTID(void) const -{ - return (reinterpret_cast(this) / sizeof(void *)) % IOTHREADS; -} - -#endif /* _WIN32 */ diff --git a/lib/base/process-windows.cpp b/lib/base/process-windows.cpp deleted file mode 100644 index f5574c4df..000000000 --- a/lib/base/process-windows.cpp +++ /dev/null @@ -1,34 +0,0 @@ -/****************************************************************************** - * Icinga 2 * - * Copyright (C) 2012-2014 Icinga Development Team (http://www.icinga.org) * - * * - * 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 Free Software Foundation; either version 2 * - * of the License, or (at your option) any later version. * - * * - * This program 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 General Public License for more details. * - * * - * You should have received a copy of the GNU General Public License * - * along with this program; if not, write to the Free Software Foundation * - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. * - ******************************************************************************/ - -#include "base/process.h" - -#ifdef _WIN32 -using namespace icinga; - -void Process::Run(const boost::function& callback) -{ - // TODO: implement - ProcessResult pr; - - if (callback) - callback(pr); -} - -#endif /* _WIN32 */ diff --git a/lib/base/process.cpp b/lib/base/process.cpp index bfc848fd8..e240be0f1 100644 --- a/lib/base/process.cpp +++ b/lib/base/process.cpp @@ -19,40 +19,125 @@ #include "i2-base.h" #include "base/process.h" +#include "base/exception.h" +#include "base/convert.h" #include "base/array.h" #include "base/objectlock.h" +#include "base/utility.h" +#include "base/initialize.h" +#include "base/logger_fwd.h" +#include "base/utility.h" +#include "base/scriptvariable.h" #include +#include +#include +#include +#include + +#ifndef _WIN32 +# include +# include + +# ifndef __APPLE__ +extern char **environ; +# else /* __APPLE__ */ +# include +# define environ (*_NSGetEnviron()) +# endif /* __APPLE__ */ +#endif /* _WIN32 */ using namespace icinga; -Process::Process(const std::vector& arguments, const Dictionary::Ptr& extraEnvironment) +#define IOTHREADS 8 + +static boost::mutex l_ProcessMutex[IOTHREADS]; +static std::map l_Processes[IOTHREADS]; +#ifdef _WIN32 +static HANDLE l_Events[IOTHREADS]; +#else /* _WIN32 */ +static int l_EventFDs[IOTHREADS][2]; +static std::map l_FDs[IOTHREADS]; +#endif /* _WIN32 */ +static boost::once_flag l_OnceFlag = BOOST_ONCE_INIT; + +INITIALIZE_ONCE(&Process::StaticInitialize); + +Process::Process(const Process::Arguments& arguments, const Dictionary::Ptr& extraEnvironment) : m_Arguments(arguments), m_ExtraEnvironment(extraEnvironment), m_Timeout(600) { } -std::vector Process::SplitCommand(const Value& command) +void Process::StaticInitialize(void) { + for (int tid = 0; tid < IOTHREADS; tid++) { +#ifdef _WIN32 + l_Events[tid] = CreateEvent(NULL, TRUE, FALSE, NULL); +#else /* _WIN32 */ +# ifdef HAVE_PIPE2 + if (pipe2(l_EventFDs[tid], O_CLOEXEC) < 0) { + BOOST_THROW_EXCEPTION(posix_error() + << boost::errinfo_api_function("pipe2") + << boost::errinfo_errno(errno)); + } +# else /* HAVE_PIPE2 */ + if (pipe(l_EventFDs[tid]) < 0) { + BOOST_THROW_EXCEPTION(posix_error() + << boost::errinfo_api_function("pipe") + << boost::errinfo_errno(errno)); + } + + Utility::SetCloExec(l_EventFDs[tid][0]); + Utility::SetCloExec(l_EventFDs[tid][1]); +# endif /* HAVE_PIPE2 */ + + Utility::SetNonBlocking(l_EventFDs[tid][0]); + Utility::SetNonBlocking(l_EventFDs[tid][1]); +#endif /* _WIN32 */ + } +} + +void Process::ThreadInitialize(void) +{ + /* Note to self: Make sure this runs _after_ we've daemonized. */ + for (int tid = 0; tid < IOTHREADS; tid++) { + boost::thread t(boost::bind(&Process::IOThreadProc, tid)); + t.detach(); + } +} + +Process::Arguments Process::PrepareCommand(const Value& command) +{ +#ifdef _WIN32 + String args; +#else /* _WIN32 */ std::vector args; +#endif /* _WIN32 */ if (command.IsObjectType()) { Array::Ptr arguments = command; ObjectLock olock(arguments); BOOST_FOREACH(const Value& argument, arguments) { +#ifdef _WIN32 + if (args != "") + args += " "; + + args += Utility::EscapeShellArg(argument); +#else /* _WIN32 */ args.push_back(argument); +#endif /* _WIN32 */ } return args; } - // TODO: implement #ifdef _WIN32 - args.push_back(command); + return command; #else /* _WIN32 */ args.push_back("sh"); args.push_back("-c"); args.push_back(command); -#endif return args; +#endif } void Process::SetTimeout(double timeout) @@ -64,3 +149,403 @@ double Process::GetTimeout(void) const { return m_Timeout; } + +void Process::IOThreadProc(int tid) +{ +#ifdef _WIN32 + HANDLE *handles = NULL; +#else /* _WIN32 */ + pollfd *pfds = NULL; +#endif /* _WIN32 */ + int count = 0; + + Utility::SetThreadName("ProcessIO"); + + for (;;) { + double now, timeout = -1; + + now = Utility::GetTime(); + + { + boost::mutex::scoped_lock lock(l_ProcessMutex[tid]); + + count = 1 + l_Processes[tid].size(); +#ifdef _WIN32 + handles = reinterpret_cast(realloc(handles, sizeof(HANDLE) * count)); + + handles[0] = l_Events[tid]; + +#else /* _WIN32 */ + pfds = reinterpret_cast(realloc(pfds, sizeof(pollfd) * count)); + + pfds[0].fd = l_EventFDs[tid][0]; + pfds[0].events = POLLIN; + pfds[0].revents = 0; +#endif /* _WIN32 */ + + int i = 1; + std::pair kv; + BOOST_FOREACH(kv, l_Processes[tid]) { +#ifdef _WIN32 + handles[i] = kv.first; +#else /* _WIN32 */ + pfds[i].fd = kv.second->m_FD; + pfds[i].events = POLLIN; + pfds[i].revents = 0; +#endif /* _WIN32 */ + + if (kv.second->m_Timeout != 0) { + double delta = kv.second->m_Timeout - (now - kv.second->m_Result.ExecutionStart); + + if (timeout == -1 || delta < timeout) + timeout = delta; + } + + i++; + } + } + + if (timeout != -1) + timeout *= 1000; + +#ifdef _WIN32 + DWORD rc = WaitForMultipleObjects(count, handles, FALSE, timeout == -1 ? INFINITE : static_cast(timeout)); +#else /* _WIN32 */ + int rc = poll(pfds, count, timeout); + + if (rc < 0) + continue; +#endif /* _WIN32 */ + + { + boost::mutex::scoped_lock lock(l_ProcessMutex[tid]); + +#ifdef _WIN32 + if (rc == WAIT_OBJECT_0) + ResetEvent(l_Events[tid]); +#else /* _WIN32 */ + if (pfds[0].revents & (POLLIN | POLLHUP | POLLERR)) { + char buffer[512]; + (void)read(l_EventFDs[tid][0], buffer, sizeof(buffer)); + } +#endif /* _WIN32 */ + + for (int i = 1; i < count; i++) { +#ifdef _WIN32 + if (rc == WAIT_OBJECT_0 + i) { +#else /* _WIN32 */ + if (pfds[i].revents & (POLLIN | POLLHUP | POLLERR)) { + std::map::iterator it2; + it2 = l_FDs[tid].find(pfds[i].fd); + + if (it2 == l_FDs[tid].end()) + continue; /* This should never happen. */ + +#endif /* _WIN32 */ + std::map::iterator it; +#ifdef _WIN32 + it = l_Processes[tid].find(handles[i]); +#else /* _WIN32 */ + it = l_Processes[tid].find(it2->second); +#endif /* _WIN32 */ + + if (it == l_Processes[tid].end()) + continue; /* This should never happen. */ + + if (!it->second->DoEvents()) { +#ifdef _WIN32 + CloseHandle(it->first); + CloseHandle(it->second->m_FD); +#else /* _WIN32 */ + l_FDs[tid].erase(it->second->m_FD); + (void)close(it->second->m_FD); +#endif /* _WIN32 */ + l_Processes[tid].erase(it); + } + } + } + } + } +} + +void Process::Run(const boost::function& callback) +{ + boost::call_once(l_OnceFlag, &Process::ThreadInitialize); + + m_Result.ExecutionStart = Utility::GetTime(); + +#ifdef _WIN32 + HANDLE outReadPipe, outWritePipe; + if (!CreatePipe(&outReadPipe, &outWritePipe, NULL, 0)) + BOOST_THROW_EXCEPTION(win32_error() + << boost::errinfo_api_function("CreatePipe") + << errinfo_win32_error(GetLastError())); + + if (!SetHandleInformation(outReadPipe, HANDLE_FLAG_INHERIT, 0)) + BOOST_THROW_EXCEPTION(win32_error() + << boost::errinfo_api_function("SetHandleInformation") + << errinfo_win32_error(GetLastError())); + + HANDLE outWritePipeDup; + if (!DuplicateHandle(GetCurrentProcess(), outWritePipe, GetCurrentProcess(), + &outWritePipeDup, 0, TRUE, DUPLICATE_SAME_ACCESS)) + BOOST_THROW_EXCEPTION(win32_error() + << boost::errinfo_api_function("DuplicateHandle") + << errinfo_win32_error(GetLastError())); + + STARTUPINFO si = {}; + si.cb = sizeof(si); + si.hStdError = outWritePipe; + si.hStdOutput = outWritePipeDup; + si.hStdInput = GetStdHandle(STD_INPUT_HANDLE); + si.dwFlags = STARTF_USESTDHANDLES; + + PROCESS_INFORMATION pi; + + char *args = new char[m_Arguments.GetLength() + 1]; + strncpy(args, m_Arguments.CStr(), m_Arguments.GetLength() + 1); + args[m_Arguments.GetLength()] = '\0'; + + if (!CreateProcess(NULL, args, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi)) { + CloseHandle(outWritePipe); + CloseHandle(outWritePipeDup); + delete args; + BOOST_THROW_EXCEPTION(win32_error() + << boost::errinfo_api_function("CreateProcess") + << errinfo_win32_error(GetLastError())); + } + + delete args; + + CloseHandle(outWritePipe); + CloseHandle(outWritePipeDup); + CloseHandle(pi.hThread); + + m_Process = pi.hProcess; + m_FD = outReadPipe; + + Log(LogDebug, "base", "Running command '" + m_Arguments + + "': PID " + Convert::ToString(pi.dwProcessId)); + +#else /* _WIN32 */ + int fds[2]; + +#ifdef HAVE_PIPE2 + if (pipe2(fds, O_CLOEXEC) < 0) { + BOOST_THROW_EXCEPTION(posix_error() + << boost::errinfo_api_function("pipe2") + << boost::errinfo_errno(errno)); + } +#else /* HAVE_PIPE2 */ + if (pipe(fds) < 0) { + BOOST_THROW_EXCEPTION(posix_error() + << boost::errinfo_api_function("pipe") + << boost::errinfo_errno(errno)); + } + + Utility::SetCloExec(fds[0]); + Utility::SetCloExec(fds[1]); +#endif /* HAVE_PIPE2 */ + + // build argv + char **argv = new char *[m_Arguments.size() + 1]; + + for (unsigned int i = 0; i < m_Arguments.size(); i++) + argv[i] = strdup(m_Arguments[i].CStr()); + + argv[m_Arguments.size()] = NULL; + + // build envp + int envc = 0; + + /* count existing environment variables */ + while (environ[envc] != NULL) + envc++; + + char **envp = new char *[envc + (m_ExtraEnvironment ? m_ExtraEnvironment->GetLength() : 0) + 1]; + + for (int i = 0; i < envc; i++) + envp[i] = strdup(environ[i]); + + if (m_ExtraEnvironment) { + ObjectLock olock(m_ExtraEnvironment); + + int index = envc; + BOOST_FOREACH(const Dictionary::Pair& kv, m_ExtraEnvironment) { + String skv = kv.first + "=" + Convert::ToString(kv.second); + envp[index] = strdup(skv.CStr()); + index++; + } + } + + envp[envc + (m_ExtraEnvironment ? m_ExtraEnvironment->GetLength() : 0)] = NULL; + + m_ExtraEnvironment.reset(); + +#ifdef HAVE_VFORK + Value use_vfork = ScriptVariable::Get("UseVfork"); + + if (use_vfork.IsEmpty() || static_cast(use_vfork)) + m_Process = vfork(); + else + m_Process = fork(); +#else /* HAVE_VFORK */ + m_Process = fork(); +#endif /* HAVE_VFORK */ + + if (m_Process < 0) { + BOOST_THROW_EXCEPTION(posix_error() + << boost::errinfo_api_function("fork") + << boost::errinfo_errno(errno)); + } + + if (m_Process == 0) { + // child process + + if (dup2(fds[1], STDOUT_FILENO) < 0 || dup2(fds[1], STDERR_FILENO) < 0) { + perror("dup2() failed."); + _exit(128); + } + + (void)close(fds[0]); + (void)close(fds[1]); + + (void)nice(5); + + if (icinga2_execvpe(argv[0], argv, envp) < 0) { + perror("execvpe() failed."); + _exit(128); + } + + _exit(128); + } + + // parent process + + Log(LogDebug, "base", "Running command '" + boost::algorithm::join(m_Arguments, " ") + + "': PID " + Convert::ToString(m_Process)); + + m_Arguments.clear(); + + // free arguments + for (int i = 0; argv[i] != NULL; i++) + free(argv[i]); + + delete[] argv; + + // free environment + for (int i = 0; envp[i] != NULL; i++) + free(envp[i]); + + delete[] envp; + + (void)close(fds[1]); + + Utility::SetNonBlocking(fds[0]); + + m_FD = fds[0]; +#endif /* _WIN32 */ + + m_Callback = callback; + + int tid = GetTID(); + + { + boost::mutex::scoped_lock lock(l_ProcessMutex[tid]); + l_Processes[tid][m_Process] = GetSelf(); +#ifndef _WIN32 + l_FDs[tid][m_FD] = m_Process; +#endif /* _WIN32 */ + } + +#ifdef _WIN32 + SetEvent(l_Events[tid]); +#else /* _WIN32 */ + (void)write(l_EventFDs[tid][1], "T", 1); +#endif /* _WIN32 */ +} + +bool Process::DoEvents(void) +{ + if (m_Timeout != 0) { + double timeout = m_Timeout - (Utility::GetTime() - m_Result.ExecutionStart); + + if (timeout < 0) { + m_OutputStream << ""; +#ifdef _WIN32 + TerminateProcess(m_Process, 1); +#else /* _WIN32 */ + kill(m_Process, SIGKILL); +#endif /* _WIN32 */ + } + } + + char buffer[512]; + for (;;) { +#ifdef _WIN32 + DWORD rc; + if (!ReadFile(m_FD, buffer, sizeof(buffer), &rc, NULL) || rc == 0) + break; +#else /* _WIN32 */ + int rc = read(m_FD, buffer, sizeof(buffer)); + + if (rc < 0 && (errno == EAGAIN || errno == EWOULDBLOCK)) + return true; + + if (rc > 0) { +#endif /* _WIN32 */ + m_OutputStream.write(buffer, rc); +#ifdef _WIN32 + return true; +#else /* _WIN32 */ + continue; + } +#endif /* _WIN32 */ + + break; + } + + String output = m_OutputStream.str(); + +#ifdef _WIN32 + WaitForSingleObject(m_Process, INFINITE); + + DWORD exitcode; + GetExitCodeProcess(m_Process, &exitcode); +#else /* _WIN32 */ + int status, exitcode; + if (waitpid(m_Process, &status, 0) != m_Process) { + BOOST_THROW_EXCEPTION(posix_error() + << boost::errinfo_api_function("waitpid") + << boost::errinfo_errno(errno)); + } + + if (WIFEXITED(status)) { + exitcode = WEXITSTATUS(status); + } + else if (WIFSIGNALED(status)) { + std::ostringstream outputbuf; + outputbuf << ""; + output = output + outputbuf.str(); + exitcode = 128; + } + else { + exitcode = 128; + } +#endif /* _WIN32 */ + + m_Result.ExecutionEnd = Utility::GetTime(); + m_Result.ExitStatus = exitcode; + m_Result.Output = output; + + if (m_Callback) + Utility::QueueAsyncCallback(boost::bind(m_Callback, m_Result)); + + return false; +} + +int Process::GetTID(void) const +{ + return (reinterpret_cast(this) / sizeof(void *)) % IOTHREADS; +} + diff --git a/lib/base/process.h b/lib/base/process.h index a7343256a..d3ea6f866 100644 --- a/lib/base/process.h +++ b/lib/base/process.h @@ -56,31 +56,39 @@ class I2_BASE_API Process : public Object public: DECLARE_PTR_TYPEDEFS(Process); +#ifdef _WIN32 + typedef String Arguments; + typedef HANDLE ProcessHandle; + typedef HANDLE ConsoleHandle; +#else /* _WIN32 */ + typedef std::vector Arguments; + typedef pid_t ProcessHandle; + typedef int ConsoleHandle; +#endif /* _WIN32 */ + static const std::deque::size_type MaxTasksPerThread = 512; - Process(const std::vector& arguments, const Dictionary::Ptr& extraEnvironment = Dictionary::Ptr()); + Process(const Arguments& arguments, const Dictionary::Ptr& extraEnvironment = Dictionary::Ptr()); void SetTimeout(double timeout); double GetTimeout(void) const; void Run(const boost::function& callback = boost::function()); - static std::vector SplitCommand(const Value& command); + static Arguments PrepareCommand(const Value& command); -#ifndef _WIN32 static void StaticInitialize(void); static void ThreadInitialize(void); -#endif /* _WIN32 */ private: - std::vector m_Arguments; + Arguments m_Arguments; Dictionary::Ptr m_ExtraEnvironment; double m_Timeout; -#ifndef _WIN32 - pid_t m_Pid; - int m_FD; + ProcessHandle m_Process; + ConsoleHandle m_FD; + std::ostringstream m_OutputStream; boost::function m_Callback; ProcessResult m_Result; @@ -88,7 +96,6 @@ private: static void IOThreadProc(int tid); bool DoEvents(void); int GetTID(void) const; -#endif /* _WIN32 */ }; } diff --git a/lib/methods/pluginchecktask.cpp b/lib/methods/pluginchecktask.cpp index 05f9cded3..144d91e66 100644 --- a/lib/methods/pluginchecktask.cpp +++ b/lib/methods/pluginchecktask.cpp @@ -70,7 +70,7 @@ void PluginCheckTask::ScriptFunc(const Checkable::Ptr& checkable, const CheckRes cr->SetCommand(command); - Process::Ptr process = make_shared(Process::SplitCommand(command), envMacros); + Process::Ptr process = make_shared(Process::PrepareCommand(command), envMacros); process->SetTimeout(commandObj->GetTimeout()); diff --git a/lib/methods/plugineventtask.cpp b/lib/methods/plugineventtask.cpp index 6ff612ba6..e91630ec5 100644 --- a/lib/methods/plugineventtask.cpp +++ b/lib/methods/plugineventtask.cpp @@ -65,7 +65,7 @@ void PluginEventTask::ScriptFunc(const Checkable::Ptr& checkable) } } - Process::Ptr process = make_shared(Process::SplitCommand(command), envMacros); + Process::Ptr process = make_shared(Process::PrepareCommand(command), envMacros); process->SetTimeout(commandObj->GetTimeout()); diff --git a/lib/methods/pluginnotificationtask.cpp b/lib/methods/pluginnotificationtask.cpp index 06055b8b5..1beb2c287 100644 --- a/lib/methods/pluginnotificationtask.cpp +++ b/lib/methods/pluginnotificationtask.cpp @@ -80,7 +80,7 @@ void PluginNotificationTask::ScriptFunc(const Notification::Ptr& notification, c } } - Process::Ptr process = make_shared(Process::SplitCommand(command), envMacros); + Process::Ptr process = make_shared(Process::PrepareCommand(command), envMacros); process->SetTimeout(commandObj->GetTimeout());