class Options : private boost::orable<Options> {
friend class Subprocess;
public:
- Options() : closeOtherFds_(false), usePath_(false) { }
+ Options()
+ : closeOtherFds_(false),
+ usePath_(false),
+ parentDeathSignal_(0) {
+ }
/**
* Change action for file descriptor fd.
*/
Options& usePath() { usePath_ = true; return *this; }
+ /**
+ * Child will receive a signal when the parent exits.
+ */
+ Options& parentDeathSignal(int sig) {
+ parentDeathSignal_ = sig;
+ return *this;
+ }
+
/**
* Helpful way to combine Options.
*/
FdMap fdActions_;
bool closeOtherFds_;
bool usePath_;
+ int parentDeathSignal_;
};
static Options pipeStdin() { return Options().stdin(PIPE); }
#include "folly/Subprocess.h"
+#include <unistd.h>
+
#include <glog/logging.h>
#include <gtest/gtest.h>
#include "folly/experimental/Gen.h"
#include "folly/experimental/FileGen.h"
#include "folly/experimental/StringGen.h"
+#include "folly/experimental/io/FsUtil.h"
using namespace folly;
EXPECT_EQ(1, proc.wait().exitStatus());
}
+TEST(ParentDeathSubprocessTest, ParentDeathSignal) {
+ // Find out where we are.
+ static constexpr size_t pathLength = 2048;
+ char buf[pathLength];
+ int r = readlink("/proc/self/exe", buf, pathLength);
+ CHECK_ERR(r >= 0);
+ buf[r] = '\0';
+
+ fs::path helper(buf);
+ helper.remove_filename();
+ helper /= "subprocess_test_parent_death_helper";
+
+ fs::path tempFile(fs::temp_directory_path() / fs::unique_path());
+
+ std::vector<std::string> args {helper.string(), tempFile.string()};
+ Subprocess proc(args);
+ // The helper gets killed by its child, see details in
+ // SubprocessTestParentDeathHelper.cpp
+ ASSERT_EQ(SIGKILL, proc.wait().killSignal());
+
+ // Now wait for the file to be created, see details in
+ // SubprocessTestParentDeathHelper.cpp
+ while (!fs::exists(tempFile)) {
+ usleep(20000); // 20ms
+ }
+
+ fs::remove(tempFile);
+}
+
TEST(PopenSubprocessTest, PopenRead) {
Subprocess proc("ls /", Subprocess::pipeStdout());
int found = 0;
--- /dev/null
+/*
+ * Copyright 2013 Facebook, Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+// This is a helper for the parentDeathSignal test in SubprocessTest.cpp.
+//
+// Basically, we create two processes, a parent and a child, and set the
+// child to receive SIGUSR1 when the parent exits. We set the child to
+// create a file when that happens. The child then kills the parent; the test
+// will verify that the file actually gets created, which means that everything
+// worked as intended.
+
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <signal.h>
+#include <unistd.h>
+
+#include <gflags/gflags.h>
+#include <glog/logging.h>
+
+#include "folly/Conv.h"
+#include "folly/Subprocess.h"
+
+using folly::Subprocess;
+
+DEFINE_bool(child, false, "");
+
+namespace {
+constexpr int kSignal = SIGUSR1;
+volatile bool caught = false;
+
+void signalHandler(int sig) {
+ if (sig != kSignal) {
+ abort();
+ }
+ caught = true;
+}
+
+} // namespace
+
+void runChild(const char* file) {
+ struct sigaction sa;
+ sa.sa_handler = signalHandler;
+ sigemptyset(&sa.sa_mask);
+ sa.sa_flags = 0;
+ CHECK_ERR(sigaction(kSignal, &sa, nullptr));
+
+ // Kill the parent, wait for our signal.
+ CHECK_ERR(kill(getppid(), SIGKILL));
+
+ while (!caught) {
+ pause();
+ }
+
+ // Signal completion by creating the file
+ CHECK_ERR(creat(file, 0600));
+}
+
+void runParent(const char* file) {
+ std::vector<std::string> args {"/proc/self/exe", "--child", file};
+ Subprocess proc(
+ args,
+ Subprocess::Options().parentDeathSignal(kSignal));
+ CHECK(proc.poll().running());
+
+ // The child will kill us.
+ for (;;) {
+ pause();
+ }
+}
+
+int main(int argc, char *argv[]) {
+ google::ParseCommandLineFlags(&argc, &argv, true);
+ CHECK_EQ(argc, 2);
+ if (FLAGS_child) {
+ runChild(argv[1]);
+ } else {
+ runParent(argv[1]);
+ }
+ return 0;
+}
+