blob: bb0a46a5288993ca1ff5b39e99d3b9bd5db0d734 [file] [log] [blame]
// Copyright 2013 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/extensions/api/messaging/native_messaging_test_util.h"
#include <memory>
#include <tuple>
#include <utility>
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/json/json_file_value_serializer.h"
#include "base/path_service.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "base/threading/thread_restrictions.h"
#include "base/values.h"
#include "build/build_config.h"
#include "chrome/common/chrome_paths.h"
#include "testing/gtest/include/gtest/gtest.h"
#if BUILDFLAG(IS_WIN)
#include "base/win/registry.h"
#endif
namespace extensions {
namespace {
void WriteTestNativeHostManifest(const base::FilePath& target_dir,
const std::string& host_name,
const base::FilePath& host_path,
bool user_level,
bool supports_native_initiated_connections) {
auto manifest = base::Value::Dict()
.Set("name", host_name)
.Set("description", "Native Messaging Echo Test")
.Set("type", "stdio")
.Set("path", host_path.AsUTF8Unsafe())
.Set("supports_native_initiated_connections",
supports_native_initiated_connections);
manifest.Set("allowed_origins",
base::Value::List().Append(base::StringPrintf(
"chrome-extension://%s/",
ScopedTestNativeMessagingHost::kExtensionId)));
base::FilePath manifest_path = target_dir.AppendASCII(host_name + ".json");
JSONFileValueSerializer serializer(manifest_path);
ASSERT_TRUE(serializer.Serialize(manifest));
#if BUILDFLAG(IS_WIN)
HKEY root_key = user_level ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE;
std::wstring key = L"SOFTWARE\\Google\\Chrome\\NativeMessagingHosts\\" +
base::UTF8ToWide(host_name);
base::win::RegKey manifest_key(
root_key, key.c_str(),
KEY_SET_VALUE | KEY_CREATE_SUB_KEY | KEY_CREATE_LINK);
ASSERT_EQ(ERROR_SUCCESS,
manifest_key.WriteValue(NULL, manifest_path.value().c_str()));
#endif
}
} // namespace
const char ScopedTestNativeMessagingHost::kHostName[] =
"com.google.chrome.test.echo";
const char ScopedTestNativeMessagingHost::kBinaryMissingHostName[] =
"com.google.chrome.test.host_binary_missing";
const char ScopedTestNativeMessagingHost::
kSupportsNativeInitiatedConnectionsHostName[] =
"com.google.chrome.test.inbound_native_echo";
const char ScopedTestNativeMessagingHost::kExtensionId[] =
"knldjmfmopnpolahpmmgbagdohdnhkik";
#if BUILDFLAG(IS_WIN)
const char ScopedTestNativeMessagingHost::kHostExeName[] =
"com.google.chrome.test.exe.echo";
#endif
ScopedTestNativeMessagingHost::ScopedTestNativeMessagingHost() = default;
void ScopedTestNativeMessagingHost::RegisterTestHost(bool user_level) {
base::ScopedAllowBlockingForTesting allow_blocking;
ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
base::FilePath test_user_data_dir;
ASSERT_TRUE(
base::PathService::Get(chrome::DIR_TEST_DATA, &test_user_data_dir));
test_user_data_dir = test_user_data_dir.AppendASCII("native_messaging")
.AppendASCII("native_hosts");
#if BUILDFLAG(IS_WIN)
HKEY root_key = user_level ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE;
ASSERT_NO_FATAL_FAILURE(registry_override_.OverrideRegistry(root_key));
#else
path_override_ = std::make_unique<base::ScopedPathOverride>(
user_level ? chrome::DIR_USER_NATIVE_MESSAGING
: chrome::DIR_NATIVE_MESSAGING,
temp_dir_.GetPath());
#endif
base::CopyFile(test_user_data_dir.AppendASCII("echo.py"),
temp_dir_.GetPath().AppendASCII("echo.py"));
#if BUILDFLAG(IS_WIN)
base::FilePath host_path = temp_dir_.GetPath().AppendASCII("echo.bat");
base::CopyFile(test_user_data_dir.AppendASCII("echo.bat"), host_path);
#endif
#if BUILDFLAG(IS_POSIX)
base::FilePath host_path = temp_dir_.GetPath().AppendASCII("echo.py");
ASSERT_TRUE(base::SetPosixFilePermissions(
host_path, base::FILE_PERMISSION_READ_BY_USER |
base::FILE_PERMISSION_WRITE_BY_USER |
base::FILE_PERMISSION_EXECUTE_BY_USER));
#endif
ASSERT_NO_FATAL_FAILURE(WriteTestNativeHostManifest(
temp_dir_.GetPath(), kHostName, host_path, user_level, false));
ASSERT_NO_FATAL_FAILURE(WriteTestNativeHostManifest(
temp_dir_.GetPath(), kBinaryMissingHostName,
test_user_data_dir.AppendASCII("missing_nm_binary.exe"), user_level,
false));
ASSERT_NO_FATAL_FAILURE(WriteTestNativeHostManifest(
temp_dir_.GetPath(), kSupportsNativeInitiatedConnectionsHostName,
host_path, user_level, true));
}
#if BUILDFLAG(IS_WIN)
// On Windows, a new codepath is used to directly launch .EXE-based Native
// Hosts.
void ScopedTestNativeMessagingHost::RegisterTestExeHost(
std::string_view filename,
bool user_level) {
base::ScopedAllowBlockingForTesting allow_blocking;
ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
base::FilePath binary_dir;
ASSERT_TRUE(base::PathService::Get(base::DIR_EXE, &binary_dir));
HKEY root_key = user_level ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE;
ASSERT_NO_FATAL_FAILURE(registry_override_.OverrideRegistry(root_key));
// Unlike in the |RegisterTestHost| case above, we must leave the Host
// .exe where it was built, because the Host will fail to run from the
// temp_dir_ if is_component_build is set for the build.
base::FilePath host_path = binary_dir.AppendASCII(filename);
ASSERT_NO_FATAL_FAILURE(WriteTestNativeHostManifest(
temp_dir_.GetPath(), kHostExeName, host_path, user_level, false));
}
#endif
ScopedTestNativeMessagingHost::~ScopedTestNativeMessagingHost() {
base::ScopedAllowBlockingForTesting allow_blocking;
std::ignore = temp_dir_.Delete();
}
} // namespace extensions