// Copyright 2019 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/apps/app_shim/mach_bootstrap_acceptor.h"
#include <bsm/libbsm.h>
#include "base/process/process_handle.h"
#include "base/run_loop.h"
#include "base/test/task_environment.h"
#include "chrome/app_shim/app_shim_controller.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace apps {
class MachBootstrapAcceptorTest : public testing::Test {
public:
// Friend accessors:
static mojo::PlatformChannelEndpoint ConnectToBrowser(
const mojo::NamedPlatformChannel::ServerName& server_name) {
return AppShimController::ConnectToBrowser(server_name);
}
static mach_port_t GetAcceptorPort(MachBootstrapAcceptor* acceptor) {
return acceptor->port();
}
static const mojo::NamedPlatformChannel::ServerName& GetAcceptorServerName(
MachBootstrapAcceptor* acceptor) {
return acceptor->server_name_;
}
private:
base::test::TaskEnvironment task_environment_;
};
class TestMachBootstrapAcceptorDelegate
: public MachBootstrapAcceptor::Delegate {
public:
explicit TestMachBootstrapAcceptorDelegate(base::OnceClosure quit_closure)
: quit_closure_(std::move(quit_closure)) {}
void OnClientConnected(mojo::PlatformChannelEndpoint endpoint,
audit_token_t audit_token) override {
endpoint_ = std::move(endpoint);
pid_ = audit_token_to_pid(audit_token);
std::move(quit_closure_).Run();
}
void OnServerChannelCreateError() override {
error_ = true;
std::move(quit_closure_).Run();
}
base::ProcessId pid() const { return pid_; }
const mojo::PlatformChannelEndpoint& endpoint() const { return endpoint_; }
bool error() const { return error_; }
private:
base::ProcessId pid_ = base::kNullProcessId;
mojo::PlatformChannelEndpoint endpoint_;
bool error_ = false;
base::OnceClosure quit_closure_;
};
TEST_F(MachBootstrapAcceptorTest, SingleRequest) {
base::RunLoop run_loop;
TestMachBootstrapAcceptorDelegate delegate(run_loop.QuitClosure());
MachBootstrapAcceptor acceptor("simplereq", &delegate);
acceptor.Start();
EXPECT_TRUE(GetAcceptorPort(&acceptor) != MACH_PORT_NULL);
mojo::PlatformChannelEndpoint endpoint =
ConnectToBrowser(GetAcceptorServerName(&acceptor));
run_loop.Run();
EXPECT_FALSE(delegate.error());
EXPECT_EQ(base::GetCurrentProcId(), delegate.pid());
// In the same process, the send and receive rights are known by the same
// Mach port name.
EXPECT_EQ(endpoint.platform_handle().GetMachReceiveRight().get(),
delegate.endpoint().platform_handle().GetMachSendRight().get());
}
TEST_F(MachBootstrapAcceptorTest, FailToRegister) {
base::RunLoop run_loop;
TestMachBootstrapAcceptorDelegate delegate(run_loop.QuitClosure());
MachBootstrapAcceptor acceptor("failtoreg", &delegate);
mojo::NamedPlatformChannel::ServerName server_name =
GetAcceptorServerName(&acceptor);
// Squat on the server name in the bootstrap server.
mojo::NamedPlatformChannel::Options server_options;
server_options.server_name = server_name;
mojo::NamedPlatformChannel server_endpoint =
mojo::NamedPlatformChannel(server_options);
// The acceptor will fail to start and reports an error.
acceptor.Start();
run_loop.Run();
EXPECT_TRUE(delegate.error());
}
} // namespace apps