blob: 86e34b8ae0f6fe6982db4931d9fc05b0d188b7ce [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/base/resource/data_pack.h"
#include <stddef.h>
#include <stdint.h>
#include <utility>
#include "base/files/file.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/files/scoped_temp_dir.h"
#include "base/path_service.h"
#include "base/strings/string_piece.h"
#include "build/build_config.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/zlib/google/compression_utils.h"
#include "ui/base/resource/data_pack_literal.h"
#include "ui/base/ui_base_paths.h"
namespace ui {
class DataPackTest
: public testing::TestWithParam<DataPack::TextEncodingType> {
public:
DataPackTest() {}
};
TEST(DataPackTest, LoadFromPath) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath data_path =
dir.GetPath().Append(FILE_PATH_LITERAL("sample.pak"));
// Dump contents into the pak file.
ASSERT_EQ(base::WriteFile(data_path, kSamplePakContentsV4, kSamplePakSizeV4),
static_cast<int>(kSamplePakSizeV4));
// Load the file through the data pack API.
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromPath(data_path));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(4));
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ("this is id 4", data);
ASSERT_TRUE(pack.HasResource(6));
ASSERT_TRUE(pack.GetStringPiece(6, &data));
EXPECT_EQ("this is id 6", data);
// Try reading zero-length data blobs, just in case.
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(0U, data.length());
ASSERT_TRUE(pack.GetStringPiece(10, &data));
EXPECT_EQ(0U, data.length());
// Try looking up an invalid key.
ASSERT_FALSE(pack.HasResource(140));
ASSERT_FALSE(pack.GetStringPiece(140, &data));
}
TEST(DataPackTest, LoadFromPathCompressed) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath data_path =
dir.GetPath().Append(FILE_PATH_LITERAL("sample.pak.gz"));
// Dump contents into a compressed pak file.
std::string compressed;
ASSERT_TRUE(compression::GzipCompress(
base::StringPiece(kSamplePakContentsV4, kSamplePakSizeV4), &compressed));
ASSERT_EQ(base::WriteFile(data_path, compressed.c_str(), compressed.length()),
static_cast<int>(compressed.length()));
// Load the file through the data pack API.
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromPath(data_path));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(4));
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ("this is id 4", data);
ASSERT_TRUE(pack.HasResource(6));
ASSERT_TRUE(pack.GetStringPiece(6, &data));
EXPECT_EQ("this is id 6", data);
// Try reading zero-length data blobs, just in case.
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(0U, data.length());
ASSERT_TRUE(pack.GetStringPiece(10, &data));
EXPECT_EQ(0U, data.length());
// Try looking up an invalid key.
ASSERT_FALSE(pack.HasResource(140));
ASSERT_FALSE(pack.GetStringPiece(140, &data));
}
TEST(DataPackTest, LoadFromFile) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath data_path =
dir.GetPath().Append(FILE_PATH_LITERAL("sample.pak"));
// Dump contents into the pak file.
ASSERT_EQ(base::WriteFile(data_path, kSamplePakContentsV4, kSamplePakSizeV4),
static_cast<int>(kSamplePakSizeV4));
base::File file(data_path, base::File::FLAG_OPEN | base::File::FLAG_READ);
ASSERT_TRUE(file.IsValid());
// Load the file through the data pack API.
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromFile(std::move(file)));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(4));
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ("this is id 4", data);
ASSERT_TRUE(pack.HasResource(6));
ASSERT_TRUE(pack.GetStringPiece(6, &data));
EXPECT_EQ("this is id 6", data);
// Try reading zero-length data blobs, just in case.
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(0U, data.length());
ASSERT_TRUE(pack.GetStringPiece(10, &data));
EXPECT_EQ(0U, data.length());
// Try looking up an invalid key.
ASSERT_FALSE(pack.HasResource(140));
ASSERT_FALSE(pack.GetStringPiece(140, &data));
}
TEST(DataPackTest, LoadFromFileRegion) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath data_path =
dir.GetPath().Append(FILE_PATH_LITERAL("sample.pak"));
// Construct a file which has a non page-aligned zero-filled header followed
// by the actual pak file content.
const char kPadding[5678] = {0};
ASSERT_EQ(static_cast<int>(sizeof(kPadding)),
base::WriteFile(data_path, kPadding, sizeof(kPadding)));
ASSERT_TRUE(
base::AppendToFile(data_path, kSamplePakContentsV4, kSamplePakSizeV4));
base::File file(data_path, base::File::FLAG_OPEN | base::File::FLAG_READ);
ASSERT_TRUE(file.IsValid());
// Load the file through the data pack API.
DataPack pack(SCALE_FACTOR_100P);
base::MemoryMappedFile::Region region = {sizeof(kPadding), kSamplePakSizeV4};
ASSERT_TRUE(pack.LoadFromFileRegion(std::move(file), region));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(4));
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ("this is id 4", data);
ASSERT_TRUE(pack.HasResource(6));
ASSERT_TRUE(pack.GetStringPiece(6, &data));
EXPECT_EQ("this is id 6", data);
// Try reading zero-length data blobs, just in case.
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(0U, data.length());
ASSERT_TRUE(pack.GetStringPiece(10, &data));
EXPECT_EQ(0U, data.length());
// Try looking up an invalid key.
ASSERT_FALSE(pack.HasResource(140));
ASSERT_FALSE(pack.GetStringPiece(140, &data));
}
TEST(DataPackTest, LoadFromBufferV4) {
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromBuffer(
base::StringPiece(kSamplePakContentsV4, kSamplePakSizeV4)));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(4));
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ("this is id 4", data);
ASSERT_TRUE(pack.HasResource(6));
ASSERT_TRUE(pack.GetStringPiece(6, &data));
EXPECT_EQ("this is id 6", data);
// Try reading zero-length data blobs, just in case.
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(0U, data.length());
ASSERT_TRUE(pack.GetStringPiece(10, &data));
EXPECT_EQ(0U, data.length());
// Try looking up an invalid key.
ASSERT_FALSE(pack.HasResource(140));
ASSERT_FALSE(pack.GetStringPiece(140, &data));
}
TEST(DataPackTest, LoadFromBufferV5) {
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromBuffer(base::StringPiece(
kSampleCompressPakContentsV5, kSampleCompressPakSizeV5)));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(4));
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ("this is id 4", data);
ASSERT_TRUE(pack.HasResource(6));
ASSERT_TRUE(pack.GetStringPiece(6, &data));
ASSERT_TRUE(pack.HasResource(8));
ASSERT_TRUE(pack.GetStringPiece(8, &data));
ASSERT_TRUE(pack.GetStringPiece(10, &data));
ASSERT_EQ("this is id 4", data);
// Try looking up an invalid key.
ASSERT_FALSE(pack.HasResource(140));
ASSERT_FALSE(pack.GetStringPiece(140, &data));
}
INSTANTIATE_TEST_SUITE_P(WriteBINARY,
DataPackTest,
::testing::Values(DataPack::BINARY));
INSTANTIATE_TEST_SUITE_P(WriteUTF8,
DataPackTest,
::testing::Values(DataPack::UTF8));
INSTANTIATE_TEST_SUITE_P(WriteUTF16,
DataPackTest,
::testing::Values(DataPack::UTF16));
TEST(DataPackTest, LoadFileWithTruncatedHeader) {
base::FilePath data_path;
ASSERT_TRUE(base::PathService::Get(UI_DIR_TEST_DATA, &data_path));
data_path = data_path.AppendASCII("data_pack_unittest/truncated-header.pak");
DataPack pack(SCALE_FACTOR_100P);
ASSERT_FALSE(pack.LoadFromPath(data_path));
}
TEST_P(DataPackTest, Write) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath file = dir.GetPath().Append(FILE_PATH_LITERAL("data.pak"));
std::string one("one");
std::string two("two");
std::string three("three");
std::string four("four");
std::string fifteen("fifteen");
std::map<uint16_t, base::StringPiece> resources;
resources.emplace(1, base::StringPiece(one));
resources.emplace(2, base::StringPiece(two));
resources.emplace(15, base::StringPiece(fifteen));
resources.emplace(3, base::StringPiece(three));
resources.emplace(4, base::StringPiece(four));
ASSERT_TRUE(DataPack::WritePack(file, resources, GetParam()));
// Now try to read the data back in.
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromPath(file));
EXPECT_EQ(pack.GetTextEncodingType(), GetParam());
base::StringPiece data;
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(one, data);
ASSERT_TRUE(pack.GetStringPiece(2, &data));
EXPECT_EQ(two, data);
ASSERT_TRUE(pack.GetStringPiece(3, &data));
EXPECT_EQ(three, data);
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ(four, data);
ASSERT_TRUE(pack.GetStringPiece(15, &data));
EXPECT_EQ(fifteen, data);
EXPECT_EQ(5U, pack.GetResourceTableSizeForTesting());
EXPECT_EQ(0U, pack.GetAliasTableSizeForTesting());
}
TEST_P(DataPackTest, WriteWithAliases) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath file = dir.GetPath().Append(FILE_PATH_LITERAL("data.pak"));
std::string one("one");
std::string two("two");
std::string three("three");
std::string four("four");
std::string fifteen("fifteen");
std::map<uint16_t, base::StringPiece> resources;
resources.emplace(1, base::StringPiece(one));
resources.emplace(2, base::StringPiece(two));
resources.emplace(15, base::StringPiece(fifteen));
resources.emplace(3, base::StringPiece(three));
resources.emplace(4, base::StringPiece(four));
resources.emplace(10, base::StringPiece(one));
resources.emplace(11, base::StringPiece(three));
ASSERT_TRUE(DataPack::WritePack(file, resources, GetParam()));
// Now try to read the data back in.
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromPath(file));
EXPECT_EQ(pack.GetTextEncodingType(), GetParam());
base::StringPiece data;
ASSERT_TRUE(pack.GetStringPiece(1, &data));
EXPECT_EQ(one, data);
ASSERT_TRUE(pack.GetStringPiece(2, &data));
EXPECT_EQ(two, data);
ASSERT_TRUE(pack.GetStringPiece(3, &data));
EXPECT_EQ(three, data);
ASSERT_TRUE(pack.GetStringPiece(4, &data));
EXPECT_EQ(four, data);
ASSERT_TRUE(pack.GetStringPiece(15, &data));
EXPECT_EQ(fifteen, data);
ASSERT_TRUE(pack.GetStringPiece(10, &data));
EXPECT_EQ(one, data);
ASSERT_TRUE(pack.GetStringPiece(11, &data));
EXPECT_EQ(three, data);
base::StringPiece data2;
ASSERT_TRUE(pack.GetStringPiece(1, &data));
ASSERT_TRUE(pack.GetStringPiece(10, &data2));
EXPECT_EQ(data.data(), data2.data());
ASSERT_TRUE(pack.GetStringPiece(3, &data));
ASSERT_TRUE(pack.GetStringPiece(11, &data2));
EXPECT_EQ(data.data(), data2.data());
EXPECT_EQ(5U, pack.GetResourceTableSizeForTesting());
EXPECT_EQ(2U, pack.GetAliasTableSizeForTesting());
}
#if defined(OS_POSIX)
TEST(DataPackTest, ModifiedWhileUsed) {
base::ScopedTempDir dir;
ASSERT_TRUE(dir.CreateUniqueTempDir());
base::FilePath data_path =
dir.GetPath().Append(FILE_PATH_LITERAL("sample.pak"));
// Dump contents into the pak file.
ASSERT_EQ(base::WriteFile(data_path, kSamplePakContentsV4, kSamplePakSizeV4),
static_cast<int>(kSamplePakSizeV4));
base::File file(data_path, base::File::FLAG_OPEN | base::File::FLAG_READ);
ASSERT_TRUE(file.IsValid());
// Load the file through the data pack API.
DataPack pack(SCALE_FACTOR_100P);
ASSERT_TRUE(pack.LoadFromFile(std::move(file)));
base::StringPiece data;
ASSERT_TRUE(pack.HasResource(10));
ASSERT_TRUE(pack.GetStringPiece(10, &data));
ASSERT_EQ(base::WriteFile(data_path, kSampleCorruptPakContents,
kSampleCorruptPakSize),
static_cast<int>(kSampleCorruptPakSize));
// Reading asset #10 should now fail as it extends past the end of the file.
ASSERT_TRUE(pack.HasResource(10));
ASSERT_FALSE(pack.GetStringPiece(10, &data));
}
#endif
} // namespace ui