blob: 0d9a8e9cf54f7a5ab9782ebd9b8e7b8e9282972b [file] [log] [blame]
/*
* Copyright (C) 2022 The Android Open Source Project
*
* 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.
*/
#include "host/commands/cvd/server.h"
#include <set>
#include <string>
#include <vector>
#include <android-base/file.h>
#include <android-base/logging.h>
#include <fruit/fruit.h>
#include "cvd_server.pb.h"
#include "common/libs/fs/shared_buf.h"
#include "common/libs/fs/shared_fd.h"
#include "common/libs/utils/environment.h"
#include "common/libs/utils/flag_parser.h"
#include "common/libs/utils/result.h"
#include "common/libs/utils/subprocess.h"
#include "host/libs/config/cuttlefish_config.h"
namespace cuttlefish {
namespace {
constexpr char kHostBugreportBin[] = "cvd_internal_host_bugreport";
constexpr char kStartBin[] = "cvd_internal_start";
constexpr char kClearBin[] = "clear_placeholder"; // Unused, runs CvdClear()
constexpr char kFleetBin[] = "fleet_placeholder"; // Unused, runs CvdFleet()
constexpr char kHelpBin[] = "help_placeholder"; // Unused, prints kHelpMessage.
constexpr char kHelpMessage[] = R"(Cuttlefish Virtual Device (CVD) CLI.
usage: cvd <command> <args>
Commands:
help Print this message.
help <command> Print help for a command.
start Start a device.
stop Stop a running device.
clear Stop all running devices and delete all instance and assembly directories.
fleet View the current fleet status.
kill-server Kill the cvd_server background process.
status Check and print the state of a running instance.
host_bugreport Capture a host bugreport, including configs, logs, and tombstones.
Args:
<command args> Each command has its own set of args. See cvd help <command>.
--clean If provided, runs cvd kill-server before the requested command.
)";
const std::map<std::string, std::string> CommandToBinaryMap = {
{"help", kHelpBin},
{"host_bugreport", kHostBugreportBin},
{"cvd_host_bugreport", kHostBugreportBin},
{"start", kStartBin},
{"launch_cvd", kStartBin},
{"status", kStatusBin},
{"cvd_status", kStatusBin},
{"stop", kStopBin},
{"stop_cvd", kStopBin},
{"clear", kClearBin},
{"fleet", kFleetBin}};
class CvdCommandHandler : public CvdServerHandler {
public:
INJECT(CvdCommandHandler(CvdServer& server)) : server_(server) {}
Result<bool> CanHandle(const RequestWithStdio& request) const {
auto invocation = ParseInvocation(request.request);
return CommandToBinaryMap.find(invocation.command) !=
CommandToBinaryMap.end();
}
Result<cvd::Response> Handle(const RequestWithStdio& request) {
CF_EXPECT(CanHandle(request));
cvd::Response response;
response.mutable_command_response();
auto invocation = ParseInvocation(request.request);
auto subcommand_bin = CommandToBinaryMap.find(invocation.command);
CF_EXPECT(subcommand_bin != CommandToBinaryMap.end());
auto bin = subcommand_bin->second;
// assembly_dir is used to possibly set CuttlefishConfig path env variable
// later. This env variable is used by subcommands when locating the config.
std::vector<Flag> flags;
std::string assembly_dir =
StringFromEnv("HOME", ".") + "/cuttlefish_assembly";
flags.emplace_back(GflagsCompatFlag("assembly_dir", assembly_dir));
// Create a copy of args before parsing, to be passed to subcommands.
auto args = invocation.arguments;
auto args_copy = invocation.arguments;
CHECK(ParseFlags(flags, invocation.arguments));
auto host_artifacts_path =
request.request.command_request().env().find("ANDROID_HOST_OUT");
if (host_artifacts_path == request.request.command_request().env().end()) {
response.mutable_status()->set_code(cvd::Status::FAILED_PRECONDITION);
response.mutable_status()->set_message(
"Missing ANDROID_HOST_OUT in client environment.");
return response;
}
if (bin == kHelpBin) {
// Handle `cvd help`
if (args.empty()) {
WriteAll(request.out, kHelpMessage);
response.mutable_status()->set_code(cvd::Status::OK);
return response;
}
// Certain commands have no detailed help text.
std::set<std::string> builtins = {"help", "clear", "kill-server"};
auto it = CommandToBinaryMap.find(args[0]);
if (it == CommandToBinaryMap.end() ||
builtins.find(args[0]) != builtins.end()) {
WriteAll(request.out, kHelpMessage);
response.mutable_status()->set_code(cvd::Status::OK);
return response;
}
// Handle `cvd help <subcommand>` by calling the subcommand with --help.
bin = it->second;
args_copy.push_back("--help");
} else if (bin == kClearBin) {
*response.mutable_status() = server_.CvdClear(request.out, request.err);
return response;
} else if (bin == kFleetBin) {
auto env_config = request.request.command_request().env().find(
kCuttlefishConfigEnvVarName);
std::string config_path;
if (env_config != request.request.command_request().env().end()) {
config_path = env_config->second;
}
*response.mutable_status() = server_.CvdFleet(request.out, config_path);
return response;
} else if (bin == kStartBin) {
// Track this assembly_dir in the fleet.
CvdServer::AssemblyInfo info;
info.host_binaries_dir = host_artifacts_path->second + "/bin/";
server_.SetAssembly(assembly_dir, info);
}
auto assembly_info = CF_EXPECT(server_.GetAssembly(assembly_dir));
Command command(assembly_info.host_binaries_dir + bin);
for (const std::string& arg : args_copy) {
command.AddParameter(arg);
}
// Set CuttlefishConfig path based on assembly dir,
// used by subcommands when locating the CuttlefishConfig.
if (request.request.command_request().env().count(
kCuttlefishConfigEnvVarName) == 0) {
auto config_path = GetCuttlefishConfigPath(assembly_dir);
if (config_path) {
command.AddEnvironmentVariable(kCuttlefishConfigEnvVarName,
*config_path);
}
}
for (auto& it : request.request.command_request().env()) {
command.AddEnvironmentVariable(it.first, it.second);
}
// Redirect stdin, stdout, stderr back to the cvd client
command.RedirectStdIO(Subprocess::StdIOChannel::kStdIn, request.in);
command.RedirectStdIO(Subprocess::StdIOChannel::kStdOut, request.out);
command.RedirectStdIO(Subprocess::StdIOChannel::kStdErr, request.err);
SubprocessOptions options;
options.ExitWithParent(false);
command.Start(options);
response.mutable_status()->set_code(cvd::Status::OK);
return response;
}
Result<void> Interrupt() override { return CF_ERR("Can't interrupt"); }
private:
CvdServer& server_;
};
} // namespace
CommandInvocation ParseInvocation(const cvd::Request& request) {
CommandInvocation invocation;
if (request.contents_case() != cvd::Request::ContentsCase::kCommandRequest) {
return invocation;
}
if (request.command_request().args_size() == 0) {
return invocation;
}
for (const std::string& arg : request.command_request().args()) {
invocation.arguments.push_back(arg);
}
invocation.arguments[0] = cpp_basename(invocation.arguments[0]);
if (invocation.arguments[0] == "cvd") {
if (invocation.arguments.size() == 1) {
// Show help if user invokes `cvd` alone.
invocation.command = "help";
invocation.arguments = {};
} else { // More arguments
invocation.command = invocation.arguments[1];
invocation.arguments.erase(invocation.arguments.begin());
invocation.arguments.erase(invocation.arguments.begin());
}
} else {
invocation.command = invocation.arguments[0];
invocation.arguments.erase(invocation.arguments.begin());
}
return invocation;
}
fruit::Component<> cvdCommandComponent() {
return fruit::createComponent()
.addMultibinding<CvdServerHandler, CvdCommandHandler>();
}
} // namespace cuttlefish