blob: 2d6594f64a73f24f49803443c0f3aa1695202983 [file] [log] [blame]
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ash/dbus/arc_crosh_service_provider.h"
#include <memory>
#include <string>
#include <utility>
#include "base/functional/bind.h"
#include "base/logging.h"
#include "base/memory/scoped_refptr.h"
#include "chromeos/ash/components/dbus/arc/arc.pb.h"
#include "chromeos/ash/experiences/arc/mojom/crosh.mojom.h"
#include "chromeos/ash/experiences/arc/session/arc_bridge_service.h"
#include "chromeos/ash/experiences/arc/session/arc_service_manager.h"
#include "components/user_manager/user_manager.h"
#include "dbus/exported_object.h"
#include "dbus/message.h"
#include "third_party/cros_system_api/dbus/service_constants.h"
namespace ash {
ArcCroshServiceProvider::ArcCroshServiceProvider() = default;
ArcCroshServiceProvider::~ArcCroshServiceProvider() = default;
void ArcCroshServiceProvider::Start(
scoped_refptr<dbus::ExportedObject> exported_object) {
exported_object->ExportMethod(
arc::crosh::kArcCroshInterfaceName, arc::crosh::kArcCroshRequest,
base::BindRepeating(&ArcCroshServiceProvider::Request,
weak_ptr_factory_.GetWeakPtr()),
base::BindOnce(&ArcCroshServiceProvider::OnExported,
weak_ptr_factory_.GetWeakPtr()));
}
void ArcCroshServiceProvider::Request(
dbus::MethodCall* method_call,
dbus::ExportedObject::ResponseSender response_sender) {
dbus::MessageReader reader(method_call);
arc::ArcShellExecutionRequest request;
if (!reader.PopArrayOfBytesAsProto(&request)) {
LOG(ERROR)
<< "Failed to parse incoming message as ArcShellExecutionRequest";
std::move(response_sender)
.Run(dbus::ErrorResponse::FromMethodCall(
method_call, DBUS_ERROR_INVALID_ARGS,
"No ArcShellExecutionRequest in message"));
return;
}
if (!request.has_command()) {
LOG(WARNING) << "ArcShellExecutionRequest.command is empty";
std::move(response_sender)
.Run(dbus::ErrorResponse::FromMethodCall(
method_call, DBUS_ERROR_INVALID_ARGS,
"command should not be empty"));
return;
}
arc::mojom::ArcShellCommand command;
switch (request.command()) {
case arc::ArcShellExecutionRequest_ArcShellCommand::
ArcShellExecutionRequest_ArcShellCommand_TOP:
command = arc::mojom::ArcShellCommand::kTop;
break;
case arc::ArcShellExecutionRequest_ArcShellCommand::
ArcShellExecutionRequest_ArcShellCommand_CPUINFO:
command = arc::mojom::ArcShellCommand::kCpuinfo;
break;
case arc::ArcShellExecutionRequest_ArcShellCommand::
ArcShellExecutionRequest_ArcShellCommand_MEMINFO:
command = arc::mojom::ArcShellCommand::kMeminfo;
break;
default:
LOG(FATAL) << "Not supported shell command requested: "
<< request.command();
}
arc::mojom::ArcShellExecutionRequestPtr mojo_request =
arc::mojom::ArcShellExecutionRequest::New(command);
if (!request.has_user_id()) {
LOG(WARNING) << "request.user_id is empty";
std::move(response_sender)
.Run(dbus::ErrorResponse::FromMethodCall(
method_call, DBUS_ERROR_INVALID_ARGS,
"user_id should not be empty"));
return;
}
user_manager::UserManager* user_manager = user_manager::UserManager::Get();
std::string requesting_user = request.user_id();
if (requesting_user != user_manager->GetPrimaryUser()->username_hash()) {
LOG(WARNING) << "Requesting user is not primary user";
std::move(response_sender)
.Run(dbus::ErrorResponse::FromMethodCall(
method_call, DBUS_ERROR_ACCESS_DENIED,
"Request from not primary user is prohibited"));
return;
}
arc::mojom::ArcShellExecutionInstance* mojo_instance =
ARC_GET_INSTANCE_FOR_METHOD(arc::ArcServiceManager::Get()
->arc_bridge_service()
->arc_shell_execution(),
Exec);
if (!mojo_instance) {
LOG(ERROR) << "Failed to get connection with ArcCroshService";
std::move(response_sender)
.Run(dbus::ErrorResponse::FromMethodCall(
method_call, DBUS_ERROR_FAILED,
"Failed to get connection with ArcCroshService. "
"You can make ARCVM running from starting ARC apps, "
"e.g. Play Store"));
return;
}
std::unique_ptr<dbus::Response> response =
dbus::Response::FromMethodCall(method_call);
mojo_instance->Exec(
std::move(mojo_request),
base::BindOnce(&ArcCroshServiceProvider::SendResponse,
weak_ptr_factory_.GetWeakPtr(), std::move(response),
std::move(response_sender)));
}
void ArcCroshServiceProvider::SendResponse(
std::unique_ptr<dbus::Response> response,
dbus::ExportedObject::ResponseSender response_sender,
arc::mojom::ArcShellExecutionResultPtr got_mojo_result) {
arc::ArcShellExecutionResult sending_dbus_result;
// Only `error` field has a value if the execution fails.
if (got_mojo_result->is_error()) {
sending_dbus_result.set_error(got_mojo_result->get_error());
} else {
sending_dbus_result.set_stdout(got_mojo_result->get_stdout());
}
dbus::MessageWriter writer(response.get());
writer.AppendProtoAsArrayOfBytes(sending_dbus_result);
std::move(response_sender).Run(std::move(response));
}
void ArcCroshServiceProvider::OnExported(const std::string& interface_name,
const std::string& method_name,
bool success) {
if (!success) {
LOG(ERROR) << "Failed to export " << interface_name << "." << method_name;
}
}
} // namespace ash