// Copyright 2018 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/ash/policy/reporting/single_arc_app_install_event_log.h"
#include <stdint.h>
#include <memory>
#include "base/files/file.h"
#include "base/files/file_path.h"
#include "base/files/scoped_temp_dir.h"
#include "chrome/browser/ash/policy/reporting/install_event_log.h"
#include "chrome/browser/ash/policy/reporting/single_install_event_log.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace em = enterprise_management;
namespace policy {
namespace {
static const char kPackageName[] = "com.example.app";
static const int64_t kTimestamp = 12345;
static const char kFileName[] = "event.log";
} // namespace
class SingleArcAppInstallEventLogTest : public testing::Test {
public:
SingleArcAppInstallEventLogTest(const SingleArcAppInstallEventLogTest&) =
delete;
SingleArcAppInstallEventLogTest& operator=(
const SingleArcAppInstallEventLogTest&) = delete;
protected:
SingleArcAppInstallEventLogTest() {}
// testing::Test:
void SetUp() override {
log_ = std::make_unique<SingleArcAppInstallEventLog>(kPackageName);
}
void VerifyHeader(bool incomplete) {
EXPECT_TRUE(report_.has_package());
EXPECT_EQ(kPackageName, report_.package());
EXPECT_TRUE(report_.has_incomplete());
EXPECT_EQ(incomplete, report_.incomplete());
}
void CreateFile() {
temp_dir_ = std::make_unique<base::ScopedTempDir>();
ASSERT_TRUE(temp_dir_->CreateUniqueTempDir());
file_ = std::make_unique<base::File>(temp_dir_->GetPath().Append(kFileName),
base::File::FLAG_CREATE_ALWAYS |
base::File::FLAG_WRITE |
base::File::FLAG_READ);
}
std::unique_ptr<SingleArcAppInstallEventLog> log_;
em::AppInstallReport report_;
std::unique_ptr<base::ScopedTempDir> temp_dir_;
std::unique_ptr<base::File> file_;
};
// Verify that the package name is returned correctly.
TEST_F(SingleArcAppInstallEventLogTest, GetPackage) {
EXPECT_EQ(kPackageName, log_->id());
}
// Do not add any log entries. Serialize the log. Verify that the serialization
// contains the the correct header data (package name, incomplete flag) and no
// log entries.
TEST_F(SingleArcAppInstallEventLogTest, SerializeEmpty) {
EXPECT_TRUE(log_->empty());
EXPECT_EQ(0, log_->size());
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
EXPECT_EQ(0, report_.logs_size());
}
// Add a log entry. Verify that the entry is serialized correctly.
TEST_F(SingleArcAppInstallEventLogTest, AddAndSerialize) {
em::AppInstallReportLogEvent event;
event.set_timestamp(kTimestamp);
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
log_->Add(event);
EXPECT_FALSE(log_->empty());
EXPECT_EQ(1, log_->size());
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(1, report_.logs_size());
std::string original_event;
event.SerializeToString(&original_event);
std::string log_event;
report_.logs(0).SerializeToString(&log_event);
EXPECT_EQ(original_event, log_event);
}
// Add 10 log entries. Verify that they are serialized correctly. Then, clear
// the serialized log entries and verify that the log becomes empty.
TEST_F(SingleArcAppInstallEventLogTest, SerializeAndClear) {
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < 10; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(10, log_->size());
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(10, report_.logs_size());
for (int i = 0; i < 10; ++i) {
EXPECT_EQ(i, report_.logs(i).timestamp());
}
log_->ClearSerialized();
EXPECT_TRUE(log_->empty());
EXPECT_EQ(0, log_->size());
report_.Clear();
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
EXPECT_EQ(0, report_.logs_size());
}
// Add 10 log entries. Serialize the log. Add 10 more log entries. Clear the
// serialized log entries. Verify that the log now contains the last 10 entries.
TEST_F(SingleArcAppInstallEventLogTest, SerializeAddAndClear) {
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < 10; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(10, log_->size());
log_->Serialize(&report_);
for (int i = 10; i < 20; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(20, log_->size());
log_->ClearSerialized();
EXPECT_FALSE(log_->empty());
EXPECT_EQ(10, log_->size());
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(10, report_.logs_size());
for (int i = 0; i < 10; ++i) {
EXPECT_EQ(i + 10, report_.logs(i).timestamp());
}
}
// Add more entries than the log has capacity for. Serialize the log. Verify
// that the serialization contains the most recent log entries and the
// incomplete flag is set. Then, clear the serialized log entries. Verify that
// the log becomes empty and the incomplete flag is unset.
TEST_F(SingleArcAppInstallEventLogTest, OverflowSerializeAndClear) {
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity + 1; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->Serialize(&report_);
VerifyHeader(true /* incomplete */);
ASSERT_EQ(SingleArcAppInstallEventLog::kLogCapacity, report_.logs_size());
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity; ++i) {
EXPECT_EQ(i + 1, report_.logs(i).timestamp());
}
log_->ClearSerialized();
EXPECT_TRUE(log_->empty());
EXPECT_EQ(0, log_->size());
report_.Clear();
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
EXPECT_EQ(0, report_.logs_size());
}
// Add more entries than the log has capacity for. Serialize the log. Add one
// more log entry. Clear the serialized log entries. Verify that the log now
// contains the most recent entry and the incomplete flag is unset.
TEST_F(SingleArcAppInstallEventLogTest, OverflowSerializeAddAndClear) {
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity + 1; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->Serialize(&report_);
event.set_timestamp(SingleArcAppInstallEventLog::kLogCapacity + 1);
log_->Add(event);
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->ClearSerialized();
EXPECT_FALSE(log_->empty());
EXPECT_EQ(1, log_->size());
report_.Clear();
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(1, report_.logs_size());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity + 1,
report_.logs(0).timestamp());
}
// Add more entries than the log has capacity for. Serialize the log. Add
// exactly as many entries as the log has capacity for. Clear the serialized log
// entries. Verify that the log now contains the most recent entries and the
// incomplete flag is unset.
TEST_F(SingleArcAppInstallEventLogTest, OverflowSerializeFillAndClear) {
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity + 1; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->Serialize(&report_);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity; ++i) {
event.set_timestamp(SingleArcAppInstallEventLog::kLogCapacity + i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->ClearSerialized();
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
report_.Clear();
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(SingleArcAppInstallEventLog::kLogCapacity, report_.logs_size());
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity; ++i) {
EXPECT_EQ(i + SingleArcAppInstallEventLog::kLogCapacity,
report_.logs(i).timestamp());
}
}
// Add more entries than the log has capacity for. Serialize the log. Add more
// entries than the log has capacity for. Clear the serialized log entries.
// Verify that the log now contains the most recent entries and the incomplete
// flag is set.
TEST_F(SingleArcAppInstallEventLogTest, OverflowSerializeOverflowAndClear) {
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity + 1; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->Serialize(&report_);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity + 1; ++i) {
event.set_timestamp(SingleArcAppInstallEventLog::kLogCapacity + i);
log_->Add(event);
}
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
log_->ClearSerialized();
EXPECT_FALSE(log_->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log_->size());
report_.Clear();
log_->Serialize(&report_);
VerifyHeader(true /* incomplete */);
ASSERT_EQ(SingleArcAppInstallEventLog::kLogCapacity, report_.logs_size());
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity; ++i) {
EXPECT_EQ(i + SingleArcAppInstallEventLog::kLogCapacity + 1,
report_.logs(i).timestamp());
}
}
// Load log from a file that is not open. Verify that the operation fails.
TEST_F(SingleArcAppInstallEventLogTest, FailLoad) {
base::File invalid_file;
std::unique_ptr<SingleArcAppInstallEventLog> log =
std::make_unique<SingleArcAppInstallEventLog>(kPackageName);
EXPECT_FALSE(SingleArcAppInstallEventLog::Load(&invalid_file, &log));
EXPECT_FALSE(log);
}
// Add a log entry. Store the log to a file that is not open. Verify that the
// operation fails and the log is not modified.
TEST_F(SingleArcAppInstallEventLogTest, FailStore) {
em::AppInstallReportLogEvent event;
event.set_timestamp(0);
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
log_->Add(event);
EXPECT_FALSE(log_->empty());
EXPECT_EQ(1, log_->size());
base::File invalid_file;
EXPECT_FALSE(log_->Store(&invalid_file));
EXPECT_EQ(kPackageName, log_->id());
EXPECT_FALSE(log_->empty());
EXPECT_EQ(1, log_->size());
log_->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(1, report_.logs_size());
EXPECT_EQ(0, report_.logs(0).timestamp());
}
// Store an empty log. Load the log. Verify that that the log contents are
// loaded correctly.
TEST_F(SingleArcAppInstallEventLogTest, StoreEmptyAndLoad) {
ASSERT_NO_FATAL_FAILURE(CreateFile());
log_->Store(file_.get());
file_->Seek(base::File::FROM_BEGIN, 0);
std::unique_ptr<SingleArcAppInstallEventLog> log;
EXPECT_TRUE(SingleArcAppInstallEventLog::Load(file_.get(), &log));
ASSERT_TRUE(log);
EXPECT_EQ(kPackageName, log->id());
EXPECT_TRUE(log->empty());
EXPECT_EQ(0, log->size());
log->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(0, report_.logs_size());
}
// Populate and store a log. Load the log. Verify that that the log contents are
// loaded correctly.
TEST_F(SingleArcAppInstallEventLogTest, StoreAndLoad) {
ASSERT_NO_FATAL_FAILURE(CreateFile());
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < 10; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
log_->Store(file_.get());
file_->Seek(base::File::FROM_BEGIN, 0);
std::unique_ptr<SingleArcAppInstallEventLog> log;
EXPECT_TRUE(SingleArcAppInstallEventLog::Load(file_.get(), &log));
ASSERT_TRUE(log);
EXPECT_EQ(kPackageName, log->id());
EXPECT_FALSE(log->empty());
EXPECT_EQ(10, log->size());
log->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(10, report_.logs_size());
for (int i = 0; i < 10; ++i) {
EXPECT_EQ(i, report_.logs(i).timestamp());
}
}
// Add more entries than the log has capacity for. Store the log. Load the log.
// Verify that the log is marked as incomplete.
TEST_F(SingleArcAppInstallEventLogTest, OverflowStoreAndLoad) {
ASSERT_NO_FATAL_FAILURE(CreateFile());
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < SingleArcAppInstallEventLog::kLogCapacity + 1; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
log_->Store(file_.get());
file_->Seek(base::File::FROM_BEGIN, 0);
std::unique_ptr<SingleArcAppInstallEventLog> log;
EXPECT_TRUE(SingleArcAppInstallEventLog::Load(file_.get(), &log));
ASSERT_TRUE(log);
EXPECT_EQ(kPackageName, log->id());
EXPECT_FALSE(log->empty());
EXPECT_EQ(SingleArcAppInstallEventLog::kLogCapacity, log->size());
log->Serialize(&report_);
VerifyHeader(true /* incomplete */);
}
// Populate and serialize a log. Store the log. Load the log. Clear serialized
// entries in the loaded log. Verify that no entries are removed.
TEST_F(SingleArcAppInstallEventLogTest, SerializeStoreLoadAndClear) {
ASSERT_NO_FATAL_FAILURE(CreateFile());
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < 10; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
log_->Serialize(&report_);
log_->Store(file_.get());
file_->Seek(base::File::FROM_BEGIN, 0);
std::unique_ptr<SingleArcAppInstallEventLog> log;
EXPECT_TRUE(SingleArcAppInstallEventLog::Load(file_.get(), &log));
ASSERT_TRUE(log);
EXPECT_EQ(kPackageName, log->id());
EXPECT_FALSE(log->empty());
EXPECT_EQ(10, log->size());
log->ClearSerialized();
EXPECT_FALSE(log->empty());
EXPECT_EQ(10, log->size());
report_.Clear();
log->Serialize(&report_);
VerifyHeader(false /* incomplete */);
ASSERT_EQ(10, report_.logs_size());
for (int i = 0; i < 10; ++i) {
EXPECT_EQ(i, report_.logs(i).timestamp());
}
}
// Add 20 log entries. Store the log. Truncate the file to the length of a log
// containing 10 log entries plus one byte. Load the log. Verify that the log
// contains the first 10 log entries and is marked as incomplete.
TEST_F(SingleArcAppInstallEventLogTest, LoadTruncated) {
ASSERT_NO_FATAL_FAILURE(CreateFile());
em::AppInstallReportLogEvent event;
event.set_event_type(em::AppInstallReportLogEvent::SUCCESS);
for (int i = 0; i < 10; ++i) {
event.set_timestamp(i);
log_->Add(event);
}
log_->Store(file_.get());
file_->Seek(base::File::FROM_BEGIN, 0);
const ssize_t size = file_->GetLength();
for (int i = 0; i < 10; ++i) {
event.set_timestamp(i + 10);
log_->Add(event);
}
log_->Store(file_.get());
file_->Seek(base::File::FROM_BEGIN, 0);
file_->SetLength(size + 1);
std::unique_ptr<SingleArcAppInstallEventLog> log;
EXPECT_FALSE(SingleArcAppInstallEventLog::Load(file_.get(), &log));
ASSERT_TRUE(log);
EXPECT_EQ(kPackageName, log->id());
EXPECT_FALSE(log->empty());
EXPECT_EQ(10, log->size());
report_.Clear();
log->Serialize(&report_);
VerifyHeader(true /* incomplete */);
ASSERT_EQ(10, report_.logs_size());
for (int i = 0; i < 10; ++i) {
EXPECT_EQ(i, report_.logs(i).timestamp());
}
}
} // namespace policy