/*
 *  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 <base/files/file_util.h>
#include <gtest/gtest.h>

#include <filesystem>

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 ::testing::Test {
protected:
  void SetUp() override {
    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)); }
};

TEST_F(ConfigTest, config_find) {
  std::unique_ptr<config_t> 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_t> 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_t> 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_t> config = config_new_empty();
  EXPECT_TRUE(config.get() != NULL);
}

TEST_F(ConfigTest, config_new_no_file) {
  std::unique_ptr<config_t> config = config_new("/meow");
  EXPECT_TRUE(config.get() == NULL);
}

TEST_F(ConfigTest, config_new) {
  std::unique_ptr<config_t> config = config_new(CONFIG_FILE);
  EXPECT_TRUE(config.get() != NULL);
}

TEST_F(ConfigTest, config_new_clone) {
  std::unique_ptr<config_t> config = config_new(CONFIG_FILE);
  std::unique_ptr<config_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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_t> 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));
}
