blob: a3adb37e56055912947d2270360e10a4d1199dca [file] [log] [blame]
// Copyright 2019 The ChromiumOS Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "vm_tools/garcon/ansible_playbook_application.h"
#include <errno.h>
#include <fcntl.h>
#include <map>
#include <sstream>
#include <signal.h>
#include <sys/types.h>
#include <unistd.h>
#include <utility>
#include <vector>
#include <base/files/file_path.h>
#include <base/files/file_util.h>
#include <base/functional/bind.h>
#include <base/location.h>
#include <base/logging.h>
#include <base/posix/safe_strerror.h>
#include <base/synchronization/waitable_event.h>
#include <base/task/single_thread_task_runner.h>
#include "vm_tools/common/spawn_util.h"
namespace vm_tools {
namespace garcon {
namespace {
constexpr char kStdoutCallbackEnv[] = "ANSIBLE_STDOUT_CALLBACK";
constexpr char kDefaultCallbackPluginPathEnv[] = "ANSIBLE_CALLBACK_PLUGINS";
constexpr char kStdoutCallbackName[] = "garcon";
constexpr char kDefaultCallbackPluginPath[] =
"/usr/share/ansible/plugins/callback";
// How long we should wait for a ansible-playbook process to finish.
constexpr base::TimeDelta kAnsibleProcessTimeout = base::Hours(1);
bool CreatePipe(base::ScopedFD* read_fd,
base::ScopedFD* write_fd,
std::string* error_msg) {
int fds[2];
if (pipe2(fds, O_CLOEXEC) < 0) {
*error_msg =
"Failed to open target process pipe: " + base::safe_strerror(errno);
return false;
}
read_fd->reset(fds[0]);
write_fd->reset(fds[1]);
return true;
}
} // namespace
AnsiblePlaybookApplication::AnsiblePlaybookApplication()
: task_runner_(base::SingleThreadTaskRunner::GetCurrentDefault()),
weak_ptr_factory_(this) {}
void AnsiblePlaybookApplication::AddObserver(Observer* observer) {
observers_.AddObserver(observer);
}
void AnsiblePlaybookApplication::RemoveObserver(Observer* observer) {
observers_.RemoveObserver(observer);
}
base::FilePath AnsiblePlaybookApplication::CreateAnsiblePlaybookFile(
const std::string& playbook, std::string* error_msg) {
base::FilePath ansible_dir;
bool success = base::CreateNewTempDirectory("", &ansible_dir);
if (!success) {
*error_msg = "Failed to create directory for ansible playbook file";
return base::FilePath();
}
const base::FilePath ansible_playbook_file_path =
ansible_dir.Append("playbook.yaml");
base::File ansible_playbook_file(
ansible_playbook_file_path,
base::File::FLAG_CREATE_ALWAYS | base::File::FLAG_WRITE);
if (!ansible_playbook_file.created()) {
*error_msg = "Failed to create file for Ansible playbook";
return base::FilePath();
}
if (!ansible_playbook_file.IsValid()) {
*error_msg = "Failed to create valid file for Ansible playbook";
return base::FilePath();
}
int bytes = ansible_playbook_file.WriteAtCurrentPos(playbook.c_str(),
playbook.length());
if (bytes != playbook.length()) {
*error_msg = "Failed to write Ansible playbook content to file";
return base::FilePath();
}
return ansible_playbook_file_path;
}
bool AnsiblePlaybookApplication::ExecuteAnsiblePlaybook(
const base::FilePath& ansible_playbook_file_path, std::string* error_msg) {
std::vector<std::string> argv{"ansible-playbook",
"--become",
"--connection=local",
"--inventory",
"127.0.0.1,",
ansible_playbook_file_path.value(),
"-e",
"ansible_python_interpreter=/usr/bin/python3"};
std::map<std::string, std::string> env;
env[kStdoutCallbackEnv] = kStdoutCallbackName;
env[kDefaultCallbackPluginPathEnv] = kDefaultCallbackPluginPath;
// Set child's process stdout and stderr to write end of pipes.
int stdio_fd[] = {-1, -1, -1};
if (!CreatePipe(&read_stdout_, &write_stdout_, error_msg)) {
return false;
}
if (!CreatePipe(&read_stderr_, &write_stderr_, error_msg)) {
return false;
}
stdio_fd[STDOUT_FILENO] = write_stdout_.get();
stdio_fd[STDERR_FILENO] = write_stderr_.get();
base::WaitableEvent event(base::WaitableEvent::ResetPolicy::AUTOMATIC,
base::WaitableEvent::InitialState::NOT_SIGNALED);
bool success = task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&AnsiblePlaybookApplication::SetUpStdIOWatchers,
weak_ptr_factory_.GetWeakPtr(), &event, error_msg));
event.Wait();
if (!success) {
*error_msg = "Failed to post task to set up ansible stdio watchers";
return false;
}
if (!error_msg->empty()) {
return false;
}
pid_t spawned_pid;
if (!Spawn(std::move(argv), std::move(env), "", stdio_fd, &spawned_pid)) {
*error_msg = "Failed to spawn ansible-playbook process";
return false;
}
// As we rely on ansible process to finish and close fds, we set up a timeout
// after which process is killed.
task_runner_->PostDelayedTask(
FROM_HERE,
base::BindOnce(&AnsiblePlaybookApplication::KillAnsibleProcess,
weak_ptr_factory_.GetWeakPtr(), spawned_pid),
kAnsibleProcessTimeout);
ClearWriteFDs();
return true;
}
void AnsiblePlaybookApplication::SetUpStdIOWatchers(base::WaitableEvent* event,
std::string* error_msg) {
stdout_watcher_ = base::FileDescriptorWatcher::WatchReadable(
read_stdout_.get(),
base::BindRepeating(&AnsiblePlaybookApplication::OnStdoutReadable,
weak_ptr_factory_.GetWeakPtr()));
if (!stdout_watcher_) {
*error_msg = "Failed to set watcher for ansible-playbook stdout";
event->Signal();
return;
}
stderr_watcher_ = base::FileDescriptorWatcher::WatchReadable(
read_stderr_.get(),
base::BindRepeating(&AnsiblePlaybookApplication::OnStderrReadable,
weak_ptr_factory_.GetWeakPtr()));
if (!stderr_watcher_) {
*error_msg = "Failed to set watcher for ansible-playbook stderr";
event->Signal();
return;
}
event->Signal();
return;
}
void AnsiblePlaybookApplication::OnStdoutReadable() {
char buffer[1000];
ssize_t count = read(read_stdout_.get(), buffer, sizeof(buffer));
if (count <= 0) {
stdout_watcher_.reset();
OnStdIOProcessed(false /*is_stderr*/);
return;
}
stdout_.write(buffer, count);
int index = 0;
std::vector<std::string> lines;
for (int i = 0; i < count; i++) {
if (buffer[i] == '\n') {
lines.push_back(std::string(buffer, index, i));
index = i;
}
}
if (index != count)
lines.push_back(std::string(buffer, index, count));
for (auto& observer : observers_) {
observer.OnApplyAnsiblePlaybookProgress(lines);
}
}
void AnsiblePlaybookApplication::OnStderrReadable() {
char buffer[1000];
int index = 0;
std::vector<std::string> lines;
ssize_t count = read(read_stderr_.get(), buffer, sizeof(buffer));
if (count <= 0) {
stderr_watcher_.reset();
OnStdIOProcessed(true /*is_stderr*/);
return;
}
stderr_.write(buffer, count);
for (int i = 0; i < count; i++) {
if (buffer[i] == '\n') {
lines.push_back(std::string(buffer, index, i));
index = i;
}
}
if (index != count)
lines.push_back(std::string(buffer, index, count));
for (auto& observer : observers_) {
observer.OnApplyAnsiblePlaybookProgress(lines);
}
}
void AnsiblePlaybookApplication::OnStdIOProcessed(bool is_stderr) {
if (is_stderr)
is_stderr_finished_ = true;
else
is_stdout_finished_ = true;
if (is_stderr_finished_ && is_stdout_finished_) {
std::string failure_reason;
bool success = GetPlaybookApplicationResult(&failure_reason);
for (auto& observer : observers_)
observer.OnApplyAnsiblePlaybookCompletion(success, failure_reason);
}
}
bool AnsiblePlaybookApplication::GetPlaybookApplicationResult(
std::string* failure_reason) {
const std::string stdout = stdout_.str();
const std::string stderr = stderr_.str();
const std::string execution_info =
"Ansible playbook application stdout:\n" + stdout + "\n" +
"Ansible playbook application stderr:\n" + stderr + "\n";
if (stdout.find("MESSAGE TO GARCON: TASK_FAILED") != std::string::npos) {
LOG(INFO) << "Some tasks failed during container configuration";
*failure_reason = execution_info;
return false;
}
if (!stderr.empty()) {
*failure_reason = execution_info;
return false;
}
return true;
}
void AnsiblePlaybookApplication::ClearWriteFDs() {
write_stdout_.reset();
write_stderr_.reset();
}
void AnsiblePlaybookApplication::KillAnsibleProcess(pid_t pid) {
if (kill(pid, SIGTERM) < 0) {
LOG(ERROR) << "Failed to kill ansible process: "
<< base::safe_strerror(errno);
}
for (auto& observer : observers_)
observer.OnApplyAnsiblePlaybookCompletion(false /*success*/,
"ansible process timed out");
}
} // namespace garcon
} // namespace vm_tools