blob: 155c425b9758f06ead02c64b6f42a980cbb51d33 [file] [log] [blame]
// Copyright 2014 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 "storage/browser/test/sandbox_database_test_helper.h"
#include <stdint.h>
#include <algorithm>
#include <functional>
#include <limits>
#include <vector>
#include "base/files/file.h"
#include "base/files/file_enumerator.h"
#include "base/files/file_util.h"
#include "storage/common/fileapi/file_system_util.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/leveldatabase/leveldb_chrome.h"
using storage::FilePathToString;
namespace content {
void CorruptDatabase(const base::FilePath& db_path,
leveldb::FileType type,
ptrdiff_t offset,
size_t size) {
base::FileEnumerator file_enum(db_path, false /* not recursive */,
base::FileEnumerator::DIRECTORIES | base::FileEnumerator::FILES);
base::FilePath file_path;
base::FilePath picked_file_path;
uint64_t picked_file_number = std::numeric_limits<uint64_t>::max();
while (!(file_path = file_enum.Next()).empty()) {
uint64_t number = std::numeric_limits<uint64_t>::max();
leveldb::FileType file_type;
EXPECT_TRUE(leveldb_chrome::ParseFileName(
FilePathToString(file_path.BaseName()), &number, &file_type));
if (file_type == type &&
(picked_file_number == std::numeric_limits<uint64_t>::max() ||
picked_file_number < number)) {
picked_file_path = file_path;
picked_file_number = number;
}
}
EXPECT_FALSE(picked_file_path.empty());
EXPECT_NE(std::numeric_limits<uint64_t>::max(), picked_file_number);
base::File file(picked_file_path,
base::File::FLAG_OPEN | base::File::FLAG_READ |
base::File::FLAG_WRITE);
ASSERT_TRUE(file.IsValid());
EXPECT_FALSE(file.created());
base::File::Info file_info;
EXPECT_TRUE(file.GetInfo(&file_info));
if (offset < 0)
offset += file_info.size;
EXPECT_GE(offset, 0);
EXPECT_LE(offset, file_info.size);
size = std::min(size, static_cast<size_t>(file_info.size - offset));
std::vector<char> buf(size);
int read_size = file.Read(offset, buf.data(), buf.size());
EXPECT_LT(0, read_size);
EXPECT_GE(buf.size(), static_cast<size_t>(read_size));
buf.resize(read_size);
std::transform(buf.begin(), buf.end(), buf.begin(),
std::logical_not<char>());
int written_size = file.Write(offset, buf.data(), buf.size());
EXPECT_GT(written_size, 0);
EXPECT_EQ(buf.size(), static_cast<size_t>(written_size));
}
void DeleteDatabaseFile(const base::FilePath& db_path,
leveldb::FileType type) {
base::FileEnumerator file_enum(db_path, false /* not recursive */,
base::FileEnumerator::DIRECTORIES | base::FileEnumerator::FILES);
base::FilePath file_path;
while (!(file_path = file_enum.Next()).empty()) {
uint64_t number = std::numeric_limits<uint64_t>::max();
leveldb::FileType file_type;
EXPECT_TRUE(leveldb_chrome::ParseFileName(
FilePathToString(file_path.BaseName()), &number, &file_type));
if (file_type == type) {
base::DeleteFile(file_path, false);
// We may have multiple files for the same type, so don't break here.
}
}
}
} // namespace content