Added /backup/containers/localdir/encrypted unit test

This commit is contained in:
sfc-gh-tclinkenbeard 2021-06-25 22:33:26 -07:00
parent f5aa3df917
commit 1afae7623b
6 changed files with 63 additions and 30 deletions

View File

@ -697,9 +697,9 @@ CSimpleOpt::SOption g_rgRestoreOptions[] = {
{ OPT_DEVHELP, "--dev-help", SO_NONE }, { OPT_DEVHELP, "--dev-help", SO_NONE },
{ OPT_BLOB_CREDENTIALS, "--blob_credentials", SO_REQ_SEP }, { OPT_BLOB_CREDENTIALS, "--blob_credentials", SO_REQ_SEP },
{ OPT_INCREMENTALONLY, "--incremental", SO_NONE }, { OPT_INCREMENTALONLY, "--incremental", SO_NONE },
{ OPT_ENCRYPTION_KEY_FILE, "--encryption_key_file", SO_REQ_SEP },
{ OPT_RESTORE_BEGIN_VERSION, "--begin_version", SO_REQ_SEP }, { OPT_RESTORE_BEGIN_VERSION, "--begin_version", SO_REQ_SEP },
{ OPT_RESTORE_INCONSISTENT_SNAPSHOT_ONLY, "--inconsistent_snapshot_only", SO_NONE }, { OPT_RESTORE_INCONSISTENT_SNAPSHOT_ONLY, "--inconsistent_snapshot_only", SO_NONE },
{ OPT_ENCRYPTION_KEY_FILE, "--encryption_key_file", SO_REQ_SEP },
#ifndef TLS_DISABLED #ifndef TLS_DISABLED
TLS_OPTION_FLAGS TLS_OPTION_FLAGS
#endif #endif

View File

@ -1466,7 +1466,7 @@ ACTOR Future<Void> writeAndVerifyFile(Reference<IBackupContainer> c, Reference<I
state Reference<IAsyncFile> inputFile = wait(c->readFile(f->getFileName())); state Reference<IAsyncFile> inputFile = wait(c->readFile(f->getFileName()));
int64_t fileSize = wait(inputFile->size()); int64_t fileSize = wait(inputFile->size());
ASSERT(size == fileSize); ASSERT_EQ(size, fileSize);
if (size > 0) { if (size > 0) {
state Standalone<VectorRef<uint8_t>> buf; state Standalone<VectorRef<uint8_t>> buf;
buf.resize(buf.arena(), fileSize); buf.resize(buf.arena(), fileSize);
@ -1506,12 +1506,28 @@ ACTOR static Future<Void> testWriteSnapshotFile(Reference<IBackupFile> file, Key
return Void(); return Void();
} }
ACTOR static Future<Void> testBackupContainer(std::string url) { ACTOR Future<Void> createTestEncryptionKeyFile(std::string filename) {
state Reference<IAsyncFile> keyFile = wait(IAsyncFileSystem::filesystem()->open(
filename,
IAsyncFile::OPEN_ATOMIC_WRITE_AND_CREATE | IAsyncFile::OPEN_READWRITE | IAsyncFile::OPEN_CREATE,
0600));
std::array<uint8_t, 16> testKey;
generateRandomData(testKey.data(), testKey.size());
keyFile->write(testKey.data(), testKey.size(), 0);
wait(keyFile->sync());
return Void();
}
ACTOR Future<Void> testBackupContainer(std::string url, Optional<std::string> encryptionKeyFileName) {
state FlowLock lock(100e6); state FlowLock lock(100e6);
if (encryptionKeyFileName.present()) {
wait(createTestEncryptionKeyFile(encryptionKeyFileName.get()));
}
printf("BackupContainerTest URL %s\n", url.c_str()); printf("BackupContainerTest URL %s\n", url.c_str());
state Reference<IBackupContainer> c = IBackupContainer::openContainer(url); state Reference<IBackupContainer> c = IBackupContainer::openContainer(url, encryptionKeyFileName);
// Make sure container doesn't exist, then create it. // Make sure container doesn't exist, then create it.
try { try {
@ -1655,22 +1671,25 @@ ACTOR static Future<Void> testBackupContainer(std::string url) {
return Void(); return Void();
} }
TEST_CASE("/backup/containers/localdir") { TEST_CASE("/backup/containers/localdir/unencrypted") {
if (g_network->isSimulated()) wait(testBackupContainer(format("file://%s/fdb_backups/%llx", params.getDataDir().c_str(), timer_int()), {}));
wait(testBackupContainer(format("file://simfdb/backups/%llx", timer_int())));
else
wait(testBackupContainer(format("file:///private/tmp/fdb_backups/%llx", timer_int())));
return Void(); return Void();
}; }
TEST_CASE("/backup/containers/localdir/encrypted") {
wait(testBackupContainer(format("file://%s/fdb_backups/%llx", params.getDataDir().c_str(), timer_int()),
format("%s/test_encryption_key", params.getDataDir().c_str())));
return Void();
}
TEST_CASE("/backup/containers/url") { TEST_CASE("/backup/containers/url") {
if (!g_network->isSimulated()) { if (!g_network->isSimulated()) {
const char* url = getenv("FDB_TEST_BACKUP_URL"); const char* url = getenv("FDB_TEST_BACKUP_URL");
ASSERT(url != nullptr); ASSERT(url != nullptr);
wait(testBackupContainer(url)); wait(testBackupContainer(url, {}));
} }
return Void(); return Void();
}; }
TEST_CASE("/backup/containers_list") { TEST_CASE("/backup/containers_list") {
if (!g_network->isSimulated()) { if (!g_network->isSimulated()) {
@ -1683,7 +1702,7 @@ TEST_CASE("/backup/containers_list") {
} }
} }
return Void(); return Void();
}; }
TEST_CASE("/backup/time") { TEST_CASE("/backup/time") {
// test formatTime() // test formatTime()

View File

@ -134,13 +134,10 @@ std::string BackupContainerLocalDirectory::getURLFormat() {
ACTOR static Future<Void> readEncryptionKey(std::string encryptionKeyFileName) { ACTOR static Future<Void> readEncryptionKey(std::string encryptionKeyFileName) {
state Reference<IAsyncFile> keyFile = wait(IAsyncFileSystem::filesystem()->open(encryptionKeyFileName, 0x0, 0400)); state Reference<IAsyncFile> keyFile = wait(IAsyncFileSystem::filesystem()->open(encryptionKeyFileName, 0x0, 0400));
int64_t fileSize = wait(keyFile->size());
// TODO: Use new error code and avoid hard-coding expected size
if (fileSize != 16) {
throw internal_error();
}
state std::array<uint8_t, 16> key; state std::array<uint8_t, 16> key;
wait(success(keyFile->read(key.data(), key.size(), 0))); int bytesRead = wait(keyFile->read(key.data(), key.size(), 0));
// TODO: Throw new error (fail gracefully)
ASSERT_EQ(bytesRead, key.size());
StreamCipher::Key::initializeKey(std::move(key)); StreamCipher::Key::initializeKey(std::move(key));
return Void(); return Void();
} }
@ -216,6 +213,10 @@ Future<std::vector<std::string>> BackupContainerLocalDirectory::listURLs(const s
} }
Future<Void> BackupContainerLocalDirectory::create() { Future<Void> BackupContainerLocalDirectory::create() {
if (usesEncryption()) {
return encryptionSetupFuture;
}
// TODO: Update this comment:
// Nothing should be done here because create() can be called by any process working with the container URL, // Nothing should be done here because create() can be called by any process working with the container URL,
// such as fdbbackup. Since "local directory" containers are by definition local to the machine they are // such as fdbbackup. Since "local directory" containers are by definition local to the machine they are
// accessed from, the container's creation (in this case the creation of a directory) must be ensured prior to // accessed from, the container's creation (in this case the creation of a directory) must be ensured prior to
@ -284,8 +285,8 @@ Future<Reference<IAsyncFile>> BackupContainerLocalDirectory::readFile(const std:
} }
Future<Reference<IBackupFile>> BackupContainerLocalDirectory::writeFile(const std::string& path) { Future<Reference<IBackupFile>> BackupContainerLocalDirectory::writeFile(const std::string& path) {
int flags = IAsyncFile::OPEN_NO_AIO | IAsyncFile::OPEN_UNCACHED | IAsyncFile::OPEN_CREATE | IAsyncFile::OPEN_ATOMIC_WRITE_AND_CREATE | int flags = IAsyncFile::OPEN_NO_AIO | IAsyncFile::OPEN_UNCACHED | IAsyncFile::OPEN_CREATE |
IAsyncFile::OPEN_READWRITE; IAsyncFile::OPEN_ATOMIC_WRITE_AND_CREATE | IAsyncFile::OPEN_READWRITE;
if (usesEncryption()) { if (usesEncryption()) {
flags |= IAsyncFile::OPEN_ENCRYPTED; flags |= IAsyncFile::OPEN_ENCRYPTED;
} }

View File

@ -30,7 +30,10 @@ public:
// the filename. // the filename.
static auto getFirstBlockIV(const std::string& filename) { static auto getFirstBlockIV(const std::string& filename) {
StreamCipher::IV iv; StreamCipher::IV iv;
auto hash = XXH3_128bits(filename.c_str(), filename.size()); auto salt = basename(filename);
auto pos = salt.find('.');
salt = salt.substr(0, pos);
auto hash = XXH3_128bits(salt.c_str(), salt.size());
auto high = reinterpret_cast<unsigned char*>(&hash.high64); auto high = reinterpret_cast<unsigned char*>(&hash.high64);
auto low = reinterpret_cast<unsigned char*>(&hash.low64); auto low = reinterpret_cast<unsigned char*>(&hash.low64);
std::copy(high, high + 8, &iv[0]); std::copy(high, high + 8, &iv[0]);
@ -67,7 +70,6 @@ public:
self->readBuffers.insert(block, _plaintext); self->readBuffers.insert(block, _plaintext);
plaintext = _plaintext; plaintext = _plaintext;
} }
ASSERT(plaintext.size() == FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE);
auto start = (block == firstBlock) ? plaintext.begin() + (offset % FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE) auto start = (block == firstBlock) ? plaintext.begin() + (offset % FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE)
: plaintext.begin(); : plaintext.begin();
auto end = (block == lastBlock) auto end = (block == lastBlock)
@ -86,7 +88,7 @@ public:
ACTOR static Future<Void> write(AsyncFileEncrypted* self, void const* data, int length, int64_t offset) { ACTOR static Future<Void> write(AsyncFileEncrypted* self, void const* data, int length, int64_t offset) {
ASSERT(self->canWrite); ASSERT(self->canWrite);
// All writes must append to the end of the file: // All writes must append to the end of the file:
ASSERT(offset == self->currentBlock * FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE + self->offsetInBlock); ASSERT_EQ(offset, self->currentBlock * FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE + self->offsetInBlock);
state unsigned char const* input = reinterpret_cast<unsigned char const*>(data); state unsigned char const* input = reinterpret_cast<unsigned char const*>(data);
while (length > 0) { while (length > 0) {
const auto chunkSize = std::min(length, FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE - self->offsetInBlock); const auto chunkSize = std::min(length, FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE - self->offsetInBlock);
@ -156,11 +158,13 @@ Future<Void> AsyncFileEncrypted::zeroRange(int64_t offset, int64_t length) {
} }
Future<Void> AsyncFileEncrypted::truncate(int64_t size) { Future<Void> AsyncFileEncrypted::truncate(int64_t size) {
ASSERT(false); // TODO: Not yet implemented // FIXME: Not yet implemented
ASSERT(canWrite);
return Void(); return Void();
} }
Future<Void> AsyncFileEncrypted::sync() { Future<Void> AsyncFileEncrypted::sync() {
ASSERT(canWrite);
return AsyncFileEncryptedImpl::sync(this); return AsyncFileEncryptedImpl::sync(this);
} }
@ -169,7 +173,7 @@ Future<Void> AsyncFileEncrypted::flush() {
} }
Future<int64_t> AsyncFileEncrypted::size() const { Future<int64_t> AsyncFileEncrypted::size() const {
return currentBlock * FLOW_KNOBS->ENCRYPTION_BLOCK_SIZE + offsetInBlock; return file->size();
} }
std::string AsyncFileEncrypted::getFilename() const { std::string AsyncFileEncrypted::getFilename() const {

View File

@ -57,6 +57,7 @@ class AsyncFileEncrypted : public IAsyncFile, public ReferenceCounted<AsyncFileE
uint16_t currentBlock{ 0 }; uint16_t currentBlock{ 0 };
int offsetInBlock{ 0 }; int offsetInBlock{ 0 };
std::vector<unsigned char> writeBuffer; std::vector<unsigned char> writeBuffer;
Future<Void> initialize();
public: public:
AsyncFileEncrypted(Reference<IAsyncFile>, bool canWrite); AsyncFileEncrypted(Reference<IAsyncFile>, bool canWrite);

View File

@ -39,6 +39,7 @@ struct UnitTestWorkload : TestWorkload {
std::string testPattern; std::string testPattern;
int testRunLimit; int testRunLimit;
UnitTestParameters testParams; UnitTestParameters testParams;
bool cleanupAfterTests;
PerfIntCounter testsAvailable, testsExecuted, testsFailed; PerfIntCounter testsAvailable, testsExecuted, testsFailed;
PerfDoubleCounter totalWallTime, totalSimTime; PerfDoubleCounter totalWallTime, totalSimTime;
@ -48,9 +49,14 @@ struct UnitTestWorkload : TestWorkload {
testsFailed("Test Cases Failed"), totalWallTime("Total wall clock time (s)"), testsFailed("Test Cases Failed"), totalWallTime("Total wall clock time (s)"),
totalSimTime("Total flow time (s)") { totalSimTime("Total flow time (s)") {
enabled = !clientId; // only do this on the "first" client enabled = !clientId; // only do this on the "first" client
testPattern = getOption(options, LiteralStringRef("testsMatching"), Value()).toString(); testPattern = getOption(options, "testsMatching"_sr, Value()).toString();
testRunLimit = getOption(options, LiteralStringRef("maxTestCases"), -1); testRunLimit = getOption(options, "maxTestCases"_sr, -1);
testParams.setDataDir(getOption(options, LiteralStringRef("dataDir"), "simfdb/unittests/"_sr).toString()); if (g_network->isSimulated()) {
testParams.setDataDir(getOption(options, "dataDir"_sr, "simfdb/unittests/"_sr).toString());
} else {
testParams.setDataDir(getOption(options, "dataDir"_sr, "/private/tmp/"_sr).toString());
}
cleanupAfterTests = getOption(options, "cleanupAfterTests"_sr, true);
// Consume all remaining options as testParams which the unit test can access // Consume all remaining options as testParams which the unit test can access
for (auto& kv : options) { for (auto& kv : options) {
@ -121,7 +127,9 @@ struct UnitTestWorkload : TestWorkload {
++self->testsFailed; ++self->testsFailed;
result = e; result = e;
} }
platform::eraseDirectoryRecursive(self->testParams.getDataDir()); if (self->cleanupAfterTests) {
platform::eraseDirectoryRecursive(self->testParams.getDataDir());
}
++self->testsExecuted; ++self->testsExecuted;
double wallTime = timer() - start_timer; double wallTime = timer() - start_timer;
double simTime = now() - start_now; double simTime = now() - start_now;