/*
* Copyright 2020 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 "osi/include/config.h"
#include
#include
#include
#include "AllocationTestHarness.h"
static const std::filesystem::path kConfigFile =
std::filesystem::temp_directory_path() / "config_test.conf";
static const char* CONFIG_FILE = kConfigFile.c_str();
static const char CONFIG_FILE_CONTENT[] =
" \n\
first_key=value \n\
\n\
# Device ID (DID) configuration \n\
[DID] \n\
\n\
# Record Number: 1, 2 or 3 - maximum of 3 records \n\
recordNumber = 1 \n\
\n\
# Primary Record - true or false (default) \n\
# There can be only one primary record \n\
primaryRecord = true \n\
\n\
# Vendor ID '0xFFFF' indicates no Device ID Service Record is present in the device \n\
# 0x000F = Broadcom Corporation (default) \n\
#vendorId = 0x000F \n\
\n\
# Vendor ID Source \n\
# 0x0001 = Bluetooth SIG assigned Device ID Vendor ID value (default) \n\
# 0x0002 = USB Implementer's Forum assigned Device ID Vendor ID value \n\
#vendorIdSource = 0x0001 \n\
\n\
# Product ID & Product Version \n\
# Per spec DID v1.3 0xJJMN for version is interpreted as JJ.M.N \n\
# JJ: major version number, M: minor version number, N: sub-minor version number \n\
# For example: 1200, v14.3.6 \n\
productId = 0x1200 \n\
version = 0x1111 \n\
\n\
# Optional attributes \n\
#clientExecutableURL = \n\
#serviceDescription = \n\
#documentationURL = \n\
\n\
# Additional optional DID records. Bluedroid supports up to 3 records. \n\
[DID] \n\
[DID] \n\
version = 0x1436 \n\
\n\
HiSyncId = 18446744073709551615 \n\
HiSyncId2 = 15001900 \n\
";
class ConfigTest : public AllocationTestHarness {
protected:
void SetUp() override {
AllocationTestHarness::SetUp();
FILE* fp = fopen(CONFIG_FILE, "wt");
ASSERT_NE(fp, nullptr);
ASSERT_EQ(fwrite(CONFIG_FILE_CONTENT, 1, sizeof(CONFIG_FILE_CONTENT), fp),
sizeof(CONFIG_FILE_CONTENT));
ASSERT_EQ(fclose(fp), 0);
}
void TearDown() override {
EXPECT_TRUE(std::filesystem::remove(kConfigFile));
AllocationTestHarness::TearDown();
}
};
TEST_F(ConfigTest, config_find) {
std::unique_ptr config = config_new(CONFIG_FILE);
ASSERT_NE(config, nullptr);
EXPECT_TRUE(config->Has("DID"));
auto section_iter = config->Find("DID");
ASSERT_NE(section_iter, config->sections.end());
EXPECT_FALSE(config->Has("random"));
EXPECT_EQ(config->Find("random"), config->sections.end());
}
TEST_F(ConfigTest, section_find) {
std::unique_ptr config = config_new(CONFIG_FILE);
ASSERT_NE(config, nullptr);
EXPECT_TRUE(config->Has("DID"));
auto section_iter = config->Find("DID");
ASSERT_NE(section_iter, config->sections.end());
EXPECT_EQ(section_iter->name, "DID");
EXPECT_TRUE(section_iter->Has("version"));
auto entry_iter = section_iter->Find("version");
ASSERT_NE(entry_iter, section_iter->entries.end());
EXPECT_EQ(entry_iter->key, "version");
EXPECT_EQ(entry_iter->value, "0x1436");
EXPECT_EQ(section_iter->Find("random"), section_iter->entries.end());
EXPECT_FALSE(section_iter->Has("random"));
}
TEST_F(ConfigTest, section_set) {
std::unique_ptr config = config_new(CONFIG_FILE);
ASSERT_NE(config, nullptr);
EXPECT_TRUE(config->Has("DID"));
auto section_iter = config->Find("DID");
ASSERT_NE(section_iter, config->sections.end());
EXPECT_EQ(section_iter->name, "DID");
EXPECT_FALSE(section_iter->Has("random"));
section_iter->Set("random", "foo");
EXPECT_TRUE(section_iter->Has("random"));
auto entry_iter = section_iter->Find("random");
ASSERT_NE(entry_iter, section_iter->entries.end());
EXPECT_EQ(entry_iter->key, "random");
EXPECT_EQ(entry_iter->value, "foo");
section_iter->Set("random", "bar");
EXPECT_EQ(entry_iter->value, "bar");
entry_iter = section_iter->Find("random");
ASSERT_NE(entry_iter, section_iter->entries.end());
EXPECT_EQ(entry_iter->value, "bar");
}
TEST_F(ConfigTest, config_new_empty) {
std::unique_ptr config = config_new_empty();
EXPECT_TRUE(config.get() != NULL);
}
TEST_F(ConfigTest, config_new_no_file) {
std::unique_ptr config = config_new("/meow");
EXPECT_TRUE(config.get() == NULL);
}
TEST_F(ConfigTest, config_new) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_TRUE(config.get() != NULL);
}
TEST_F(ConfigTest, config_new_clone) {
std::unique_ptr config = config_new(CONFIG_FILE);
std::unique_ptr clone = config_new_clone(*config);
config_set_string(clone.get(), CONFIG_DEFAULT_SECTION, "first_key",
"not_value");
std::string one = std::string("one");
EXPECT_STRNE(
config_get_string(*config, CONFIG_DEFAULT_SECTION, "first_key", &one)
->c_str(),
config_get_string(*clone, CONFIG_DEFAULT_SECTION, "first_key", &one)
->c_str());
}
TEST_F(ConfigTest, config_has_section) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_TRUE(config_has_section(*config, "DID"));
}
TEST_F(ConfigTest, config_has_key_in_default_section) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_TRUE(config_has_key(*config, CONFIG_DEFAULT_SECTION, "first_key"));
EXPECT_STREQ(
config_get_string(*config, CONFIG_DEFAULT_SECTION, "first_key", nullptr)
->c_str(),
"value");
}
TEST_F(ConfigTest, config_has_keys) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_TRUE(config_has_key(*config, "DID", "recordNumber"));
EXPECT_TRUE(config_has_key(*config, "DID", "primaryRecord"));
EXPECT_TRUE(config_has_key(*config, "DID", "productId"));
EXPECT_TRUE(config_has_key(*config, "DID", "version"));
}
TEST_F(ConfigTest, config_no_bad_keys) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_FALSE(config_has_key(*config, "DID_BAD", "primaryRecord"));
EXPECT_FALSE(config_has_key(*config, "DID", "primaryRecord_BAD"));
EXPECT_FALSE(
config_has_key(*config, CONFIG_DEFAULT_SECTION, "primaryRecord"));
}
TEST_F(ConfigTest, config_get_int_version) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_EQ(config_get_int(*config, "DID", "version", 0), 0x1436);
}
TEST_F(ConfigTest, config_get_int_default) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_EQ(config_get_int(*config, "DID", "primaryRecord", 123), 123);
}
TEST_F(ConfigTest, config_get_uint64) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_EQ(config_get_uint64(*config, "DID", "HiSyncId", 0),
0xFFFFFFFFFFFFFFFF);
EXPECT_EQ(config_get_uint64(*config, "DID", "HiSyncId2", 0),
uint64_t(15001900));
}
TEST_F(ConfigTest, config_get_uint64_default) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_EQ(config_get_uint64(*config, "DID", "primaryRecord", 123),
uint64_t(123));
}
TEST_F(ConfigTest, config_remove_section) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_TRUE(config_remove_section(config.get(), "DID"));
EXPECT_FALSE(config_has_section(*config, "DID"));
EXPECT_FALSE(config_has_key(*config, "DID", "productId"));
}
TEST_F(ConfigTest, config_remove_section_missing) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_FALSE(config_remove_section(config.get(), "not a section"));
}
TEST_F(ConfigTest, config_remove_key) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_EQ(config_get_int(*config, "DID", "productId", 999), 0x1200);
EXPECT_TRUE(config_remove_key(config.get(), "DID", "productId"));
EXPECT_FALSE(config_has_key(*config, "DID", "productId"));
}
TEST_F(ConfigTest, config_remove_key_missing) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_EQ(config_get_int(*config, "DID", "productId", 999), 0x1200);
EXPECT_TRUE(config_remove_key(config.get(), "DID", "productId"));
EXPECT_EQ(config_get_int(*config, "DID", "productId", 999), 999);
}
TEST_F(ConfigTest, config_save_basic) {
std::unique_ptr config = config_new(CONFIG_FILE);
EXPECT_TRUE(config_save(*config, CONFIG_FILE));
}
TEST_F(ConfigTest, checksum_read) {
auto tmp_dir = std::filesystem::temp_directory_path();
auto filename = tmp_dir / "test.checksum";
std::string checksum = "0x1234";
base::FilePath file_path(filename.string());
EXPECT_EQ(base::WriteFile(file_path, checksum.data(), checksum.size()),
(int)checksum.size());
EXPECT_EQ(checksum_read(filename.c_str()), checksum.c_str());
EXPECT_TRUE(std::filesystem::remove(filename));
}
TEST_F(ConfigTest, checksum_save) {
auto tmp_dir = std::filesystem::temp_directory_path();
auto filename = tmp_dir / "test.checksum";
std::string checksum = "0x1234";
base::FilePath file_path(filename.string());
EXPECT_TRUE(checksum_save(checksum, filename));
EXPECT_TRUE(base::PathExists(file_path));
EXPECT_TRUE(std::filesystem::remove(filename));
}