summaryrefslogtreecommitdiff
path: root/test/storage/database.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'test/storage/database.cpp')
-rw-r--r--test/storage/database.cpp391
1 files changed, 391 insertions, 0 deletions
diff --git a/test/storage/database.cpp b/test/storage/database.cpp
new file mode 100644
index 0000000000..1a2b618a57
--- /dev/null
+++ b/test/storage/database.cpp
@@ -0,0 +1,391 @@
+#include "storage.hpp"
+#include "../fixtures/fixture_log_observer.hpp"
+
+#include <future>
+
+#include <mbgl/storage/default/sqlite_cache.hpp>
+#include <mbgl/storage/resource.hpp>
+#include <mbgl/storage/response.hpp>
+#include <mbgl/util/io.hpp>
+
+#include <sqlite3.h>
+
+class ScopedTest {
+public:
+ ScopedTest(std::function<void()> destructor_) : destructor(destructor_) {}
+ ScopedTest() {}
+
+ void finish() {
+ promise.set_value();
+ }
+
+ ~ScopedTest() {
+ promise.get_future().get();
+ if (destructor) {
+ destructor();
+ }
+ }
+
+private:
+ const std::function<void()> destructor;
+ std::promise<void> promise;
+};
+
+TEST_F(Storage, DatabaseDoesNotExist) {
+ using namespace mbgl;
+
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+
+ ScopedTest test([&] {
+ auto observer = Log::removeObserver();
+ EXPECT_EQ(1ul, dynamic_cast<FixtureLogObserver*>(observer.get())->count({ EventSeverity::Error, Event::Database, 14, "unable to open database file" }));
+ });
+
+ SQLiteCache cache("test/fixtures/404/cache.db");
+
+ cache.get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ test.finish();
+ });
+}
+
+void createDir(const char* name) {
+ const int ret = mkdir(name, 0755);
+ if (ret == -1) {
+ ASSERT_EQ(EEXIST, errno);
+ } else {
+ ASSERT_EQ(0, ret);
+ }
+}
+
+void deleteFile(const char* name) {
+ const int ret = unlink(name);
+ if (ret == -1) {
+ ASSERT_EQ(ENOENT, errno);
+ } else {
+ ASSERT_EQ(0, ret);
+ }
+}
+
+
+void writeFile(const char* name, const std::string& data) {
+ mbgl::util::write_file(name, data);
+}
+
+
+TEST_F(Storage, DatabaseCreate) {
+ using namespace mbgl;
+
+ createDir("test/fixtures/database");
+ deleteFile("test/fixtures/database/cache.db");
+
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+
+ ScopedTest test([&] {
+ Log::removeObserver();
+ });
+
+ SQLiteCache cache("test/fixtures/database/cache.db");
+
+ cache.get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ test.finish();
+ });
+}
+
+class FileLock {
+public:
+ FileLock(const std::string& path) {
+ const int err = sqlite3_open_v2(path.c_str(), &db, SQLITE_OPEN_CREATE | SQLITE_OPEN_READWRITE, nullptr);
+ if (err != SQLITE_OK) {
+ throw std::runtime_error("Could not open db");
+ }
+ lock();
+ }
+
+ void lock() {
+ assert(!locked);
+ const int err = sqlite3_exec(db, "begin exclusive transaction", nullptr, nullptr, nullptr);
+ if (err != SQLITE_OK) {
+ throw std::runtime_error("Could not lock db");
+ }
+ locked = true;
+ }
+
+ void unlock() {
+ assert(locked);
+ const int err = sqlite3_exec(db, "commit", nullptr, nullptr, nullptr);
+ if (err != SQLITE_OK) {
+ throw std::runtime_error("Could not unlock db");
+ }
+ locked = false;
+ }
+
+ ~FileLock() {
+ if (locked) {
+ unlock();
+ }
+ }
+
+private:
+ sqlite3* db;
+ bool locked = false;
+};
+
+TEST_F(Storage, DatabaseLockedRead) {
+ using namespace mbgl;
+
+ // Create a locked file.
+ createDir("test/fixtures/database");
+ deleteFile("test/fixtures/database/locked.db");
+ FileLock guard("test/fixtures/database/locked.db");
+
+ auto cache = util::make_unique<SQLiteCache>("test/fixtures/database/locked.db");
+
+ std::promise<void> promise;
+
+ {
+ // First request should fail.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ // Make sure that we got a few "database locked" errors
+ auto observer = Log::removeObserver();
+ auto flo = dynamic_cast<FixtureLogObserver*>(observer.get());
+ EXPECT_EQ(2ul, flo->count({ EventSeverity::Error, Event::Database, 5, "database is locked" }));
+ }
+
+ // Then, unlock the file and try again.
+ guard.unlock();
+
+ {
+ // First, try getting a file (the cache value should not exist).
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ // Make sure that we got a no errors
+ Log::removeObserver();
+ }
+
+ // Explicitly delete the Cache now.
+ cache.reset();
+}
+
+
+
+TEST_F(Storage, DatabaseLockedWrite) {
+ using namespace mbgl;
+
+ // Create a locked file.
+ createDir("test/fixtures/database");
+ deleteFile("test/fixtures/database/locked.db");
+ FileLock guard("test/fixtures/database/locked.db");
+
+ auto cache = util::make_unique<SQLiteCache>("test/fixtures/database/locked.db");
+
+ std::promise<void> promise;
+
+ {
+ // Adds a file (which should fail).
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+ auto response = std::make_shared<Response>();
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Full);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ auto observer = Log::removeObserver();
+ auto flo = dynamic_cast<FixtureLogObserver*>(observer.get());
+ EXPECT_EQ(4ul, flo->count({ EventSeverity::Error, Event::Database, 5, "database is locked" }));
+ }
+
+ // Then, unlock the file and try again.
+ guard.unlock();
+
+ {
+ // Then, set a file and obtain it again.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+
+ auto response = std::make_shared<Response>();
+ response->data = "Demo";
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Full);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_NE(nullptr, res.get());
+ EXPECT_EQ("Demo", res->data);
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ // Make sure that we got a no errors
+ Log::removeObserver();
+ }
+
+ // Explicitly delete the Cache now.
+ cache.reset();
+}
+
+
+
+
+TEST_F(Storage, DatabaseLockedRefresh) {
+ using namespace mbgl;
+
+ // Create a locked file.
+ createDir("test/fixtures/database");
+ deleteFile("test/fixtures/database/locked.db");
+
+ auto cache = util::make_unique<SQLiteCache>("test/fixtures/database/locked.db");
+
+ // Then, lock the file and try again.
+ FileLock guard("test/fixtures/database/locked.db");
+
+ std::promise<void> promise;
+
+ {
+ // Adds a file.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+ auto response = std::make_shared<Response>();
+ response->data = "Demo";
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Full);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ auto observer = Log::removeObserver();
+ auto flo = dynamic_cast<FixtureLogObserver*>(observer.get());
+ EXPECT_EQ(4ul, flo->count({ EventSeverity::Error, Event::Database, 5, "database is locked" }));
+ }
+
+ {
+ // Then, try to refresh it.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+
+ auto response = std::make_shared<Response>();
+ response->data = "Demo";
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Refresh);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_EQ(nullptr, res.get());
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ // Make sure that we got the right errors.
+ auto observer = Log::removeObserver();
+ auto flo = dynamic_cast<FixtureLogObserver*>(observer.get());
+ EXPECT_EQ(4ul, flo->count({ EventSeverity::Error, Event::Database, 5, "database is locked" }));
+ }
+
+ // Explicitly delete the Cache now.
+ cache.reset();
+}
+
+
+
+TEST_F(Storage, DatabaseDeleted) {
+ using namespace mbgl;
+
+ // Create a locked file.
+ createDir("test/fixtures/database");
+ deleteFile("test/fixtures/database/locked.db");
+
+ auto cache = util::make_unique<SQLiteCache>("test/fixtures/database/locked.db");
+
+ std::promise<void> promise;
+
+ {
+ // Adds a file.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+ auto response = std::make_shared<Response>();
+ response->data = "Demo";
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Full);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_NE(nullptr, res.get());
+ EXPECT_EQ("Demo", res->data);
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ Log::removeObserver();
+ }
+
+ deleteFile("test/fixtures/database/locked.db");
+
+ {
+ // Adds a file.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+ auto response = std::make_shared<Response>();
+ response->data = "Demo";
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Full);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_NE(nullptr, res.get());
+ EXPECT_EQ("Demo", res->data);
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ auto observer = Log::removeObserver();
+ auto flo = dynamic_cast<FixtureLogObserver*>(observer.get());
+ EXPECT_EQ(1ul, flo->count({ EventSeverity::Error, Event::Database, 8, "attempt to write a readonly database" }));
+ }
+
+ // Explicitly delete the Cache now.
+ cache.reset();
+}
+
+
+
+TEST_F(Storage, DatabaseInvalid) {
+ using namespace mbgl;
+
+ // Create a locked file.
+ createDir("test/fixtures/database");
+ deleteFile("test/fixtures/database/invalid.db");
+ writeFile("test/fixtures/database/invalid.db", "this is an invalid file");
+
+ auto cache = util::make_unique<SQLiteCache>("test/fixtures/database/invalid.db");
+
+ std::promise<void> promise;
+
+ {
+ // Adds a file.
+ Log::setObserver(util::make_unique<FixtureLogObserver>());
+ promise = {};
+ auto response = std::make_shared<Response>();
+ response->data = "Demo";
+ cache->put({ Resource::Unknown, "mapbox://test" }, response, FileCache::Hint::Full);
+ cache->get({ Resource::Unknown, "mapbox://test" }, [&] (std::unique_ptr<Response> res) {
+ EXPECT_NE(nullptr, res.get());
+ EXPECT_EQ("Demo", res->data);
+ promise.set_value();
+ });
+ promise.get_future().get();
+
+ auto observer = Log::removeObserver();
+ auto flo = dynamic_cast<FixtureLogObserver*>(observer.get());
+ EXPECT_EQ(1ul, flo->count({ EventSeverity::Warning, Event::Database, -1, "Trashing invalid database" }));
+ }
+
+ // Explicitly delete the Cache now.
+ cache.reset();
+}