summaryrefslogtreecommitdiff
path: root/test/storage/offline_database.test.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'test/storage/offline_database.test.cpp')
-rw-r--r--test/storage/offline_database.test.cpp592
1 files changed, 440 insertions, 152 deletions
diff --git a/test/storage/offline_database.test.cpp b/test/storage/offline_database.test.cpp
index 000f24e1cd..de40d8caf7 100644
--- a/test/storage/offline_database.test.cpp
+++ b/test/storage/offline_database.test.cpp
@@ -1,5 +1,6 @@
#include <mbgl/test/util.hpp>
#include <mbgl/test/fixture_log_observer.hpp>
+#include <mbgl/test/sqlite3_test_fs.hpp>
#include <mbgl/storage/offline_database.hpp>
#include <mbgl/storage/resource.hpp>
@@ -13,51 +14,235 @@
using namespace std::literals::string_literals;
using namespace mbgl;
+using mapbox::sqlite::ResultCode;
static constexpr const char* filename = "test/fixtures/offline_database/offline.db";
+#ifndef __QT__ // Qt doesn't expose the ability to register virtual file system handlers.
+static constexpr const char* filename_test_fs = "file:test/fixtures/offline_database/offline.db?vfs=test_fs";
+#endif
-TEST(OfflineDatabase, TEST_REQUIRES_WRITE(Create)) {
- FixtureLog log;
+static void deleteDatabaseFiles() {
+ // Delete leftover journaling files as well.
util::deleteFile(filename);
+ util::deleteFile(filename + "-wal"s);
+ util::deleteFile(filename + "-journal"s);
+}
+
+static FixtureLog::Message error(ResultCode code, const char* message) {
+ return { EventSeverity::Error, Event::Database, static_cast<int64_t>(code), message };
+}
+
+static FixtureLog::Message warning(ResultCode code, const char* message) {
+ return { EventSeverity::Warning, Event::Database, static_cast<int64_t>(code), message };
+}
+
+static int databasePageCount(const std::string& path) {
+ mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
+ mapbox::sqlite::Statement stmt{ db, "pragma page_count" };
+ mapbox::sqlite::Query query{ stmt };
+ query.run();
+ return query.get<int>(0);
+}
+
+static int databaseUserVersion(const std::string& path) {
+ mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
+ mapbox::sqlite::Statement stmt{ db, "pragma user_version" };
+ mapbox::sqlite::Query query{ stmt };
+ query.run();
+ return query.get<int>(0);
+}
+
+static std::string databaseJournalMode(const std::string& path) {
+ mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
+ mapbox::sqlite::Statement stmt{ db, "pragma journal_mode" };
+ mapbox::sqlite::Query query{ stmt };
+ query.run();
+ return query.get<std::string>(0);
+}
+static int databaseSyncMode(const std::string& path) {
+ mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
+ mapbox::sqlite::Statement stmt{ db, "pragma synchronous" };
+ mapbox::sqlite::Query query{ stmt };
+ query.run();
+ return query.get<int>(0);
+}
+
+static std::vector<std::string> databaseTableColumns(const std::string& path, const std::string& name) {
+ mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
+ const auto sql = std::string("pragma table_info(") + name + ")";
+ mapbox::sqlite::Statement stmt{ db, sql.c_str() };
+ mapbox::sqlite::Query query{ stmt };
+ std::vector<std::string> columns;
+ while (query.run()) {
+ columns.push_back(query.get<std::string>(1));
+ }
+ return columns;
+}
+
+namespace fixture {
+
+const Resource resource{ Resource::Style, "mapbox://test" };
+const Resource tile = Resource::tile("mapbox://test", 1, 0, 0, 0, Tileset::Scheme::XYZ);
+const Response response = [] {
+ Response res;
+ res.data = std::make_shared<std::string>("first");
+ return res;
+}();
+
+} // namespace
+
+TEST(OfflineDatabase, TEST_REQUIRES_WRITE(Create)) {
+ FixtureLog log;
+ deleteDatabaseFiles();
OfflineDatabase db(filename);
EXPECT_FALSE(bool(db.get({ Resource::Unknown, "mapbox://test" })));
EXPECT_EQ(0u, log.uncheckedCount());
}
+#ifndef __QT__ // Qt doesn't expose the ability to register virtual file system handlers.
+TEST(OfflineDatabase, TEST_REQUIRES_WRITE(CreateFail)) {
+ FixtureLog log;
+ deleteDatabaseFiles();
+ test::SQLite3TestFS fs;
+
+ // Opening the database will fail because our mock VFS returns a SQLITE_CANTOPEN error because
+ // it is not allowed to create the file. The OfflineDatabase object should handle this gracefully
+ // and treat it like an empty cache that can't be written to.
+ fs.allowFileCreate(false);
+ OfflineDatabase db(filename_test_fs);
+ EXPECT_EQ(1u, log.count(warning(ResultCode::CantOpen, "Can't open database: unable to open database file")));
+
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ // We can try to insert things into the cache, but since the cache database isn't open, it won't be stored.
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_EQ(std::make_pair(false, uint64_t(0)), db.put(res, fixture::response));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::CantOpen, "Can't write resource: unable to open database file")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ }
+
+ // We can also still "query" the database even though it is not open, and we will always get an empty result.
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_FALSE(bool(db.get(res)));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::CantOpen, "Can't update timestamp: unable to open database file")));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::CantOpen, "Can't read resource: unable to open database file")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ }
+
+ // Now, we're "freeing up" some space on the disk, and try to insert and query again. This time, we should
+ // be opening the datbase, creating the schema, and writing the data so that we can retrieve it again.
+ fs.allowFileCreate(true);
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_EQ(std::make_pair(true, uint64_t(5)), db.put(res, fixture::response));
+ auto result = db.get(res);
+ EXPECT_EQ(0u, log.uncheckedCount());
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
+
+ // Next, set the file system to read only mode and try to read the data again. While we can't
+ // write anymore, we should still be able to read, and the query that tries to update the last
+ // accessed timestamp may fail without crashing.
+ fs.allowFileCreate(false);
+ fs.setWriteLimit(0);
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ auto result = db.get(res);
+ EXPECT_EQ(1u, log.count(warning(ResultCode::CantOpen, "Can't update timestamp: unable to open database file")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
+ fs.setDebug(false);
+
+ // We're allowing SQLite to create a journal file, but restrict the number of bytes it
+ // can write so that it can start writing the journal file, but eventually fails during the
+ // timestamp update.
+ fs.allowFileCreate(true);
+ fs.setWriteLimit(8192);
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ auto result = db.get(res);
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Full, "Can't update timestamp: database or disk is full")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
+
+ // Lastly, we're disabling all I/O to simulate a backgrounded app that is restricted from doing
+ // any disk I/O at all.
+ fs.setWriteLimit(-1);
+ fs.allowIO(false);
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ // First, try reading.
+ auto result = db.get(res);
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't update timestamp: authorization denied")));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't read resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ EXPECT_FALSE(result);
+
+ // Now try inserting.
+ EXPECT_EQ(std::make_pair(false, uint64_t(0)), db.put(res, fixture::response));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't write resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ }
+
+ // Allow deleting the database.
+ fs.reset();
+}
+#endif // __QT__
+
TEST(OfflineDatabase, TEST_REQUIRES_WRITE(SchemaVersion)) {
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
{
mapbox::sqlite::Database db = mapbox::sqlite::Database::open(filename, mapbox::sqlite::ReadWriteCreate);
+ db.setBusyTimeout(Milliseconds(1000));
db.exec("PRAGMA user_version = 1");
}
+ {
+ OfflineDatabase db(filename);
+ }
+
+ EXPECT_EQ(6, databaseUserVersion(filename));
+
OfflineDatabase db(filename);
+ // Now try inserting and reading back to make sure we have a valid database.
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_EQ(std::make_pair(true, uint64_t(5)), db.put(res, fixture::response));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ auto result = db.get(res);
+ EXPECT_EQ(0u, log.uncheckedCount());
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
EXPECT_EQ(0u, log.uncheckedCount());
}
TEST(OfflineDatabase, TEST_REQUIRES_WRITE(Invalid)) {
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
util::write_file(filename, "this is an invalid file");
OfflineDatabase db(filename);
-
-#ifndef __QT__
- // Only non-Qt platforms are setting a logger on the SQLite object.
// Checking two possibilities for the error string because it apparently changes between SQLite versions.
- EXPECT_EQ(1u,
- log.count({ EventSeverity::Info, Event::Database, static_cast<int64_t>(mapbox::sqlite::ResultCode::NotADB),
- "statement aborts at 1: [PRAGMA user_version] file is encrypted or is not a database" }, true) +
- log.count({ EventSeverity::Info, Event::Database, static_cast<int64_t>(mapbox::sqlite::ResultCode::NotADB),
- "statement aborts at 1: [PRAGMA user_version] file is not a database" }, true));
-#endif
- EXPECT_EQ(1u, log.count({ EventSeverity::Warning, Event::Database, -1, "Removing existing incompatible offline database" }));
- EXPECT_EQ(0u, log.uncheckedCount());
+ EXPECT_EQ(1u, log.count(error(ResultCode::NotADB, "Can't open database: file is encrypted or is not a database"), true) +
+ log.count(error(ResultCode::NotADB, "Can't open database: file is not a database"), true));
+ EXPECT_EQ(1u, log.count(warning(static_cast<ResultCode>(-1), "Removing existing incompatible offline database")));
+
+ // Now try inserting and reading back to make sure we have a valid database.
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_EQ(std::make_pair(true, uint64_t(5)), db.put(res, fixture::response));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ auto result = db.get(res);
+ EXPECT_EQ(0u, log.uncheckedCount());
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
}
TEST(OfflineDatabase, PutDoesNotStoreConnectionErrors) {
@@ -118,7 +303,7 @@ TEST(OfflineDatabase, PutResource) {
TEST(OfflineDatabase, TEST_REQUIRES_WRITE(GetResourceFromOfflineRegion)) {
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
util::copyFile(filename, "test/fixtures/offline_database/satellite_test.db");
OfflineDatabase db(filename, mapbox::sqlite::ReadOnly);
@@ -228,14 +413,15 @@ TEST(OfflineDatabase, CreateRegion) {
OfflineDatabase db(":memory:");
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata {{ 1, 2, 3 }};
- OfflineRegion region = db.createRegion(definition, metadata);
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
- EXPECT_EQ(definition.styleURL, region.getDefinition().styleURL);
- EXPECT_EQ(definition.bounds, region.getDefinition().bounds);
- EXPECT_EQ(definition.minZoom, region.getDefinition().minZoom);
- EXPECT_EQ(definition.maxZoom, region.getDefinition().maxZoom);
- EXPECT_EQ(definition.pixelRatio, region.getDefinition().pixelRatio);
- EXPECT_EQ(metadata, region.getMetadata());
+ EXPECT_EQ(definition.styleURL, region->getDefinition().styleURL);
+ EXPECT_EQ(definition.bounds, region->getDefinition().bounds);
+ EXPECT_EQ(definition.minZoom, region->getDefinition().minZoom);
+ EXPECT_EQ(definition.maxZoom, region->getDefinition().maxZoom);
+ EXPECT_EQ(definition.pixelRatio, region->getDefinition().pixelRatio);
+ EXPECT_EQ(metadata, region->getMetadata());
EXPECT_EQ(0u, log.uncheckedCount());
}
@@ -245,10 +431,11 @@ TEST(OfflineDatabase, UpdateMetadata) {
OfflineDatabase db(":memory:");
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata {{ 1, 2, 3 }};
- OfflineRegion region = db.createRegion(definition, metadata);
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
OfflineRegionMetadata newmetadata {{ 4, 5, 6 }};
- db.updateMetadata(region.getID(), newmetadata);
+ db.updateMetadata(region->getID(), newmetadata);
EXPECT_EQ(db.listRegions().at(0).getMetadata(), newmetadata);
EXPECT_EQ(0u, log.uncheckedCount());
@@ -260,11 +447,12 @@ TEST(OfflineDatabase, ListRegions) {
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata {{ 1, 2, 3 }};
- OfflineRegion region = db.createRegion(definition, metadata);
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
std::vector<OfflineRegion> regions = db.listRegions();
ASSERT_EQ(1u, regions.size());
- EXPECT_EQ(region.getID(), regions.at(0).getID());
+ EXPECT_EQ(region->getID(), regions.at(0).getID());
EXPECT_EQ(definition.styleURL, regions.at(0).getDefinition().styleURL);
EXPECT_EQ(definition.bounds, regions.at(0).getDefinition().bounds);
EXPECT_EQ(definition.minZoom, regions.at(0).getDefinition().minZoom);
@@ -281,14 +469,16 @@ TEST(OfflineDatabase, GetRegionDefinition) {
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata {{ 1, 2, 3 }};
- OfflineRegion region = db.createRegion(definition, metadata);
- OfflineRegionDefinition result = db.getRegionDefinition(region.getID());
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
+ auto result = db.getRegionDefinition(region->getID());
+ ASSERT_TRUE(result);
- EXPECT_EQ(definition.styleURL, result.styleURL);
- EXPECT_EQ(definition.bounds, result.bounds);
- EXPECT_EQ(definition.minZoom, result.minZoom);
- EXPECT_EQ(definition.maxZoom, result.maxZoom);
- EXPECT_EQ(definition.pixelRatio, result.pixelRatio);
+ EXPECT_EQ(definition.styleURL, result->styleURL);
+ EXPECT_EQ(definition.bounds, result->bounds);
+ EXPECT_EQ(definition.minZoom, result->minZoom);
+ EXPECT_EQ(definition.maxZoom, result->maxZoom);
+ EXPECT_EQ(definition.pixelRatio, result->pixelRatio);
EXPECT_EQ(0u, log.uncheckedCount());
}
@@ -298,15 +488,16 @@ TEST(OfflineDatabase, DeleteRegion) {
OfflineDatabase db(":memory:");
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata {{ 1, 2, 3 }};
- OfflineRegion region = db.createRegion(definition, metadata);
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
Response response;
response.noContent = true;
- db.putRegionResource(region.getID(), Resource::style("http://example.com/"), response);
- db.putRegionResource(region.getID(), Resource::tile("http://example.com/", 1.0, 0, 0, 0, Tileset::Scheme::XYZ), response);
+ db.putRegionResource(region->getID(), Resource::style("http://example.com/"), response);
+ db.putRegionResource(region->getID(), Resource::tile("http://example.com/", 1.0, 0, 0, 0, Tileset::Scheme::XYZ), response);
- db.deleteRegion(std::move(region));
+ db.deleteRegion(std::move(*region));
ASSERT_EQ(0u, db.listRegions().size());
@@ -318,38 +509,35 @@ TEST(OfflineDatabase, CreateRegionInfiniteMaxZoom) {
OfflineDatabase db(":memory:");
OfflineRegionDefinition definition { "", LatLngBounds::world(), 0, INFINITY, 1.0 };
OfflineRegionMetadata metadata;
- OfflineRegion region = db.createRegion(definition, metadata);
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
- EXPECT_EQ(0, region.getDefinition().minZoom);
- EXPECT_EQ(INFINITY, region.getDefinition().maxZoom);
+ EXPECT_EQ(0, region->getDefinition().minZoom);
+ EXPECT_EQ(INFINITY, region->getDefinition().maxZoom);
EXPECT_EQ(0u, log.uncheckedCount());
}
TEST(OfflineDatabase, TEST_REQUIRES_WRITE(ConcurrentUse)) {
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
OfflineDatabase db1(filename);
EXPECT_EQ(0u, log.uncheckedCount());
OfflineDatabase db2(filename);
- Resource resource { Resource::Style, "http://example.com/" };
- Response response;
- response.noContent = true;
-
std::thread thread1([&] {
for (auto i = 0; i < 100; i++) {
- db1.put(resource, response);
- EXPECT_TRUE(bool(db1.get(resource)));
+ db1.put(fixture::resource, fixture::response);
+ EXPECT_TRUE(bool(db1.get(fixture::resource)));
}
});
std::thread thread2([&] {
for (auto i = 0; i < 100; i++) {
- db2.put(resource, response);
- EXPECT_TRUE(bool(db2.get(resource)));
+ db2.put(fixture::resource, fixture::response);
+ EXPECT_TRUE(bool(db2.get(fixture::resource)));
}
});
@@ -411,13 +599,14 @@ TEST(OfflineDatabase, PutRegionResourceDoesNotEvict) {
FixtureLog log;
OfflineDatabase db(":memory:", 1024 * 100);
OfflineRegionDefinition definition { "", LatLngBounds::world(), 0, INFINITY, 1.0 };
- OfflineRegion region = db.createRegion(definition, OfflineRegionMetadata());
+ auto region = db.createRegion(definition, OfflineRegionMetadata());
+ ASSERT_TRUE(region);
Response response;
response.data = randomString(1024);
for (uint32_t i = 1; i <= 100; i++) {
- db.putRegionResource(region.getID(), Resource::style("http://example.com/"s + util::toString(i)), response);
+ db.putRegionResource(region->getID(), Resource::style("http://example.com/"s + util::toString(i)), response);
}
EXPECT_TRUE(bool(db.get(Resource::style("http://example.com/1"))));
@@ -448,32 +637,36 @@ TEST(OfflineDatabase, GetRegionCompletedStatus) {
OfflineDatabase db(":memory:");
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata;
- OfflineRegion region = db.createRegion(definition, metadata);
+ auto region = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region);
- OfflineRegionStatus status1 = db.getRegionCompletedStatus(region.getID());
- EXPECT_EQ(0u, status1.completedResourceCount);
- EXPECT_EQ(0u, status1.completedResourceSize);
- EXPECT_EQ(0u, status1.completedTileCount);
- EXPECT_EQ(0u, status1.completedTileSize);
+ auto status1 = db.getRegionCompletedStatus(region->getID());
+ ASSERT_TRUE(status1);
+ EXPECT_EQ(0u, status1->completedResourceCount);
+ EXPECT_EQ(0u, status1->completedResourceSize);
+ EXPECT_EQ(0u, status1->completedTileCount);
+ EXPECT_EQ(0u, status1->completedTileSize);
Response response;
response.data = std::make_shared<std::string>("data");
- uint64_t styleSize = db.putRegionResource(region.getID(), Resource::style("http://example.com/"), response);
+ uint64_t styleSize = db.putRegionResource(region->getID(), Resource::style("http://example.com/"), response);
- OfflineRegionStatus status2 = db.getRegionCompletedStatus(region.getID());
- EXPECT_EQ(1u, status2.completedResourceCount);
- EXPECT_EQ(styleSize, status2.completedResourceSize);
- EXPECT_EQ(0u, status2.completedTileCount);
- EXPECT_EQ(0u, status2.completedTileSize);
+ auto status2 = db.getRegionCompletedStatus(region->getID());
+ ASSERT_TRUE(status2);
+ EXPECT_EQ(1u, status2->completedResourceCount);
+ EXPECT_EQ(styleSize, status2->completedResourceSize);
+ EXPECT_EQ(0u, status2->completedTileCount);
+ EXPECT_EQ(0u, status2->completedTileSize);
- uint64_t tileSize = db.putRegionResource(region.getID(), Resource::tile("http://example.com/", 1.0, 0, 0, 0, Tileset::Scheme::XYZ), response);
+ uint64_t tileSize = db.putRegionResource(region->getID(), Resource::tile("http://example.com/", 1.0, 0, 0, 0, Tileset::Scheme::XYZ), response);
- OfflineRegionStatus status3 = db.getRegionCompletedStatus(region.getID());
- EXPECT_EQ(2u, status3.completedResourceCount);
- EXPECT_EQ(styleSize + tileSize, status3.completedResourceSize);
- EXPECT_EQ(1u, status3.completedTileCount);
- EXPECT_EQ(tileSize, status3.completedTileSize);
+ auto status3 = db.getRegionCompletedStatus(region->getID());
+ ASSERT_TRUE(status3);
+ EXPECT_EQ(2u, status3->completedResourceCount);
+ EXPECT_EQ(styleSize + tileSize, status3->completedResourceSize);
+ EXPECT_EQ(1u, status3->completedTileCount);
+ EXPECT_EQ(tileSize, status3->completedTileSize);
EXPECT_EQ(0u, log.uncheckedCount());
}
@@ -482,21 +675,22 @@ TEST(OfflineDatabase, HasRegionResource) {
FixtureLog log;
OfflineDatabase db(":memory:", 1024 * 100);
OfflineRegionDefinition definition { "", LatLngBounds::world(), 0, INFINITY, 1.0 };
- OfflineRegion region = db.createRegion(definition, OfflineRegionMetadata());
+ auto region = db.createRegion(definition, OfflineRegionMetadata());
+ ASSERT_TRUE(region);
- EXPECT_FALSE(bool(db.hasRegionResource(region.getID(), Resource::style("http://example.com/1"))));
- EXPECT_FALSE(bool(db.hasRegionResource(region.getID(), Resource::style("http://example.com/20"))));
+ EXPECT_FALSE(bool(db.hasRegionResource(region->getID(), Resource::style("http://example.com/1"))));
+ EXPECT_FALSE(bool(db.hasRegionResource(region->getID(), Resource::style("http://example.com/20"))));
Response response;
response.data = randomString(1024);
for (uint32_t i = 1; i <= 100; i++) {
- db.putRegionResource(region.getID(), Resource::style("http://example.com/"s + util::toString(i)), response);
+ db.putRegionResource(region->getID(), Resource::style("http://example.com/"s + util::toString(i)), response);
}
- EXPECT_TRUE(bool(db.hasRegionResource(region.getID(), Resource::style("http://example.com/1"))));
- EXPECT_TRUE(bool(db.hasRegionResource(region.getID(), Resource::style("http://example.com/20"))));
- EXPECT_EQ(1024, *(db.hasRegionResource(region.getID(), Resource::style("http://example.com/20"))));
+ EXPECT_TRUE(bool(db.hasRegionResource(region->getID(), Resource::style("http://example.com/1"))));
+ EXPECT_TRUE(bool(db.hasRegionResource(region->getID(), Resource::style("http://example.com/20"))));
+ EXPECT_EQ(1024, *(db.hasRegionResource(region->getID(), Resource::style("http://example.com/20"))));
EXPECT_EQ(0u, log.uncheckedCount());
}
@@ -505,7 +699,8 @@ TEST(OfflineDatabase, HasRegionResourceTile) {
FixtureLog log;
OfflineDatabase db(":memory:", 1024 * 100);
OfflineRegionDefinition definition { "", LatLngBounds::world(), 0, INFINITY, 1.0 };
- OfflineRegion region = db.createRegion(definition, OfflineRegionMetadata());
+ auto region = db.createRegion(definition, OfflineRegionMetadata());
+ ASSERT_TRUE(region);
Resource resource { Resource::Tile, "http://example.com/" };
resource.tileData = Resource::TileData {
@@ -519,15 +714,16 @@ TEST(OfflineDatabase, HasRegionResourceTile) {
response.data = std::make_shared<std::string>("first");
- EXPECT_FALSE(bool(db.hasRegionResource(region.getID(), resource)));
- db.putRegionResource(region.getID(), resource, response);
- EXPECT_TRUE(bool(db.hasRegionResource(region.getID(), resource)));
- EXPECT_EQ(5, *(db.hasRegionResource(region.getID(), resource)));
+ EXPECT_FALSE(bool(db.hasRegionResource(region->getID(), resource)));
+ db.putRegionResource(region->getID(), resource, response);
+ EXPECT_TRUE(bool(db.hasRegionResource(region->getID(), resource)));
+ EXPECT_EQ(5, *(db.hasRegionResource(region->getID(), resource)));
- OfflineRegion anotherRegion = db.createRegion(definition, OfflineRegionMetadata());
- EXPECT_LT(region.getID(), anotherRegion.getID());
- EXPECT_TRUE(bool(db.hasRegionResource(anotherRegion.getID(), resource)));
- EXPECT_EQ(5, *(db.hasRegionResource(anotherRegion.getID(), resource)));
+ auto anotherRegion = db.createRegion(definition, OfflineRegionMetadata());
+ ASSERT_TRUE(anotherRegion);
+ EXPECT_LT(region->getID(), anotherRegion->getID());
+ EXPECT_TRUE(bool(db.hasRegionResource(anotherRegion->getID(), resource)));
+ EXPECT_EQ(5, *(db.hasRegionResource(anotherRegion->getID(), resource)));
EXPECT_EQ(0u, log.uncheckedCount());
@@ -539,8 +735,10 @@ TEST(OfflineDatabase, OfflineMapboxTileCount) {
OfflineRegionDefinition definition { "http://example.com/style", LatLngBounds::hull({1, 2}, {3, 4}), 5, 6, 2.0 };
OfflineRegionMetadata metadata;
- OfflineRegion region1 = db.createRegion(definition, metadata);
- OfflineRegion region2 = db.createRegion(definition, metadata);
+ auto region1 = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region1);
+ auto region2 = db.createRegion(definition, metadata);
+ ASSERT_TRUE(region2);
Resource nonMapboxTile = Resource::tile("http://example.com/", 1.0, 0, 0, 0, Tileset::Scheme::XYZ);
Resource mapboxTile1 = Resource::tile("mapbox://tiles/1", 1.0, 0, 0, 0, Tileset::Scheme::XYZ);
@@ -553,27 +751,27 @@ TEST(OfflineDatabase, OfflineMapboxTileCount) {
EXPECT_EQ(0u, db.getOfflineMapboxTileCount());
// Count stays the same after putting a non-tile resource.
- db.putRegionResource(region1.getID(), Resource::style("http://example.com/"), response);
+ db.putRegionResource(region1->getID(), Resource::style("http://example.com/"), response);
EXPECT_EQ(0u, db.getOfflineMapboxTileCount());
// Count stays the same after putting a non-Mapbox tile.
- db.putRegionResource(region1.getID(), nonMapboxTile, response);
+ db.putRegionResource(region1->getID(), nonMapboxTile, response);
EXPECT_EQ(0u, db.getOfflineMapboxTileCount());
// Count increases after putting a Mapbox tile not used by another region.
- db.putRegionResource(region1.getID(), mapboxTile1, response);
+ db.putRegionResource(region1->getID(), mapboxTile1, response);
EXPECT_EQ(1u, db.getOfflineMapboxTileCount());
// Count stays the same after putting a Mapbox tile used by another region.
- db.putRegionResource(region2.getID(), mapboxTile1, response);
+ db.putRegionResource(region2->getID(), mapboxTile1, response);
EXPECT_EQ(1u, db.getOfflineMapboxTileCount());
// Count stays the same after putting a Mapbox tile used by the same region.
- db.putRegionResource(region2.getID(), mapboxTile1, response);
+ db.putRegionResource(region2->getID(), mapboxTile1, response);
EXPECT_EQ(1u, db.getOfflineMapboxTileCount());
// Count stays the same after deleting a region when the tile is still used by another region.
- db.deleteRegion(std::move(region2));
+ db.deleteRegion(std::move(*region2));
EXPECT_EQ(1u, db.getOfflineMapboxTileCount());
// Count stays the same after the putting a non-offline Mapbox tile.
@@ -581,11 +779,11 @@ TEST(OfflineDatabase, OfflineMapboxTileCount) {
EXPECT_EQ(1u, db.getOfflineMapboxTileCount());
// Count increases after putting a pre-existing, but non-offline Mapbox tile.
- db.putRegionResource(region1.getID(), mapboxTile2, response);
+ db.putRegionResource(region1->getID(), mapboxTile2, response);
EXPECT_EQ(2u, db.getOfflineMapboxTileCount());
// Count decreases after deleting a region when the tiles are not used by other regions.
- db.deleteRegion(std::move(region1));
+ db.deleteRegion(std::move(*region1));
EXPECT_EQ(0u, db.getOfflineMapboxTileCount());
EXPECT_EQ(0u, log.uncheckedCount());
@@ -596,7 +794,8 @@ TEST(OfflineDatabase, BatchInsertion) {
FixtureLog log;
OfflineDatabase db(":memory:", 1024 * 100);
OfflineRegionDefinition definition { "", LatLngBounds::world(), 0, INFINITY, 1.0 };
- OfflineRegion region = db.createRegion(definition, OfflineRegionMetadata());
+ auto region = db.createRegion(definition, OfflineRegionMetadata());
+ ASSERT_TRUE(region);
Response response;
response.data = randomString(1024);
@@ -607,7 +806,7 @@ TEST(OfflineDatabase, BatchInsertion) {
}
OfflineRegionStatus status;
- db.putRegionResources(region.getID(), resources, status);
+ db.putRegionResources(region->getID(), resources, status);
for (uint32_t i = 1; i <= 100; i++) {
EXPECT_TRUE(bool(db.get(Resource::style("http://example.com/"s + util::toString(i)))));
@@ -621,7 +820,8 @@ TEST(OfflineDatabase, BatchInsertionMapboxTileCountExceeded) {
OfflineDatabase db(":memory:", 1024 * 100);
db.setOfflineMapboxTileCountLimit(1);
OfflineRegionDefinition definition { "", LatLngBounds::world(), 0, INFINITY, 1.0 };
- OfflineRegion region = db.createRegion(definition, OfflineRegionMetadata());
+ auto region = db.createRegion(definition, OfflineRegionMetadata());
+ ASSERT_TRUE(region);
Response response;
response.data = randomString(1024);
@@ -633,68 +833,26 @@ TEST(OfflineDatabase, BatchInsertionMapboxTileCountExceeded) {
OfflineRegionStatus status;
try {
- db.putRegionResources(region.getID(), resources, status);
+ db.putRegionResources(region->getID(), resources, status);
EXPECT_FALSE(true);
} catch (const MapboxTileLimitExceededException&) {
// Expected
}
- EXPECT_EQ(status.completedTileCount, 1u);
- EXPECT_EQ(status.completedResourceCount, 2u);
- EXPECT_EQ(db.getRegionCompletedStatus(region.getID()).completedTileCount, 1u);
- EXPECT_EQ(db.getRegionCompletedStatus(region.getID()).completedResourceCount, 2u);
+ EXPECT_EQ(0u, status.completedTileCount);
+ EXPECT_EQ(0u, status.completedResourceCount);
+ const auto completedStatus = db.getRegionCompletedStatus(region->getID());
+ ASSERT_TRUE(completedStatus);
+ EXPECT_EQ(1u, completedStatus->completedTileCount);
+ EXPECT_EQ(2u, completedStatus->completedResourceCount);
EXPECT_EQ(0u, log.uncheckedCount());
}
-static int databasePageCount(const std::string& path) {
- mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
- mapbox::sqlite::Statement stmt{ db, "pragma page_count" };
- mapbox::sqlite::Query query{ stmt };
- query.run();
- return query.get<int>(0);
-}
-
-static int databaseUserVersion(const std::string& path) {
- mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
- mapbox::sqlite::Statement stmt{ db, "pragma user_version" };
- mapbox::sqlite::Query query{ stmt };
- query.run();
- return query.get<int>(0);
-}
-
-static std::string databaseJournalMode(const std::string& path) {
- mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
- mapbox::sqlite::Statement stmt{ db, "pragma journal_mode" };
- mapbox::sqlite::Query query{ stmt };
- query.run();
- return query.get<std::string>(0);
-}
-
-static int databaseSyncMode(const std::string& path) {
- mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
- mapbox::sqlite::Statement stmt{ db, "pragma synchronous" };
- mapbox::sqlite::Query query{ stmt };
- query.run();
- return query.get<int>(0);
-}
-
-static std::vector<std::string> databaseTableColumns(const std::string& path, const std::string& name) {
- mapbox::sqlite::Database db = mapbox::sqlite::Database::open(path, mapbox::sqlite::ReadOnly);
- const auto sql = std::string("pragma table_info(") + name + ")";
- mapbox::sqlite::Statement stmt{ db, sql.c_str() };
- mapbox::sqlite::Query query{ stmt };
- std::vector<std::string> columns;
- while (query.run()) {
- columns.push_back(query.get<std::string>(1));
- }
- return columns;
-}
-
TEST(OfflineDatabase, MigrateFromV2Schema) {
// v2.db is a v2 database containing a single offline region with a small number of resources.
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
util::copyFile(filename, "test/fixtures/offline_database/v2.db");
{
@@ -715,7 +873,7 @@ TEST(OfflineDatabase, MigrateFromV2Schema) {
TEST(OfflineDatabase, MigrateFromV3Schema) {
// v3.db is a v3 database, migrated from v2.
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
util::copyFile(filename, "test/fixtures/offline_database/v3.db");
{
@@ -734,7 +892,7 @@ TEST(OfflineDatabase, MigrateFromV3Schema) {
TEST(OfflineDatabase, MigrateFromV4Schema) {
// v4.db is a v4 database, migrated from v2 & v3. This database used `journal_mode = WAL` and `synchronous = NORMAL`.
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
util::copyFile(filename, "test/fixtures/offline_database/v4.db");
{
@@ -760,7 +918,7 @@ TEST(OfflineDatabase, MigrateFromV4Schema) {
TEST(OfflineDatabase, MigrateFromV5Schema) {
// v5.db is a v5 database, migrated from v2, v3 & v4.
FixtureLog log;
- util::deleteFile(filename);
+ deleteDatabaseFiles();
util::copyFile(filename, "test/fixtures/offline_database/v5.db");
{
@@ -808,3 +966,133 @@ TEST(OfflineDatabase, DowngradeSchema) {
EXPECT_EQ(1u, log.count({ EventSeverity::Warning, Event::Database, -1, "Removing existing incompatible offline database" }));
EXPECT_EQ(0u, log.uncheckedCount());
}
+
+TEST(OfflineDatabase, CorruptDatabaseOnOpen) {
+ FixtureLog log;
+ util::deleteFile(filename);
+ util::copyFile(filename, "test/fixtures/offline_database/corrupt-immediate.db");
+
+ // This database is corrupt in a way that will prevent opening the database.
+ OfflineDatabase db(filename);
+ EXPECT_EQ(1u, log.count(error(ResultCode::Corrupt, "Can't open database: database disk image is malformed"), true));
+ EXPECT_EQ(1u, log.count(warning(static_cast<ResultCode>(-1), "Removing existing incompatible offline database")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ // Now try inserting and reading back to make sure we have a valid database.
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_EQ(std::make_pair(true, uint64_t(5)), db.put(res, fixture::response));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ auto result = db.get(res);
+ EXPECT_EQ(0u, log.uncheckedCount());
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
+}
+
+TEST(OfflineDatabase, CorruptDatabaseOnQuery) {
+ FixtureLog log;
+ util::deleteFile(filename);
+ util::copyFile(filename, "test/fixtures/offline_database/corrupt-delayed.db");
+
+ // This database is corrupt in a way that won't manifest itself until we start querying it,
+ // so just opening it will not cause an error.
+ OfflineDatabase db(filename);
+
+ // Just opening this corrupt database should not have produced an error yet, since
+ // PRAGMA user_version still succeeds with this database.
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ // The first request fails because the database is corrupt and has to be recreated.
+ EXPECT_EQ(nullopt, db.get(fixture::tile));
+ EXPECT_EQ(1u, log.count(error(ResultCode::Corrupt, "Can't read resource: database disk image is malformed"), true));
+ EXPECT_EQ(1u, log.count(warning(static_cast<ResultCode>(-1), "Removing existing incompatible offline database")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ // Now try inserting and reading back to make sure we have a valid database.
+ for (const auto& res : { fixture::resource, fixture::tile }) {
+ EXPECT_EQ(std::make_pair(true, uint64_t(5)), db.put(res, fixture::response));
+ EXPECT_EQ(0u, log.uncheckedCount());
+ auto result = db.get(res);
+ EXPECT_EQ(0u, log.uncheckedCount());
+ ASSERT_TRUE(result && result->data);
+ EXPECT_EQ("first", *result->data);
+ }
+}
+
+#ifndef __QT__ // Qt doesn't expose the ability to register virtual file system handlers.
+TEST(OfflineDatabase, TEST_REQUIRES_WRITE(DisallowedIO)) {
+ FixtureLog log;
+ deleteDatabaseFiles();
+ test::SQLite3TestFS fs;
+
+ OfflineDatabase db(filename_test_fs);
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ // First, create a region object so that we can try deleting it later.
+ OfflineTilePyramidRegionDefinition definition(
+ "mapbox://style", LatLngBounds::hull({ 37.66, -122.57 }, { 37.83, -122.32 }), 0, 8, 2);
+ auto region = db.createRegion(definition, {});
+ ASSERT_TRUE(region);
+
+ // Now forbid any type of IO on the database and test that none of the calls crashes.
+ fs.allowIO(false);
+
+ EXPECT_EQ(nullopt, db.get(fixture::resource));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't update timestamp: authorization denied")));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't read resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(std::make_pair(false, uint64_t(0)), db.put(fixture::resource, fixture::response));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't write resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ const auto regions = db.listRegions();
+ EXPECT_TRUE(regions.empty());
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't list regions: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(nullopt, db.createRegion(definition, {}));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't create region: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(nullopt, db.updateMetadata(region->getID(), {}));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't update region metadata: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(nullopt, db.getRegionResource(region->getID(), fixture::resource));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't update timestamp: authorization denied")));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't read region resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(nullopt, db.hasRegionResource(region->getID(), fixture::resource));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't query region resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(0u, db.putRegionResource(region->getID(), fixture::resource, fixture::response));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't write region resource: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ OfflineRegionStatus status;
+ db.putRegionResources(region->getID(), { std::make_tuple(fixture::resource, fixture::response) }, status);
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't write region resources: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(nullopt, db.getRegionDefinition(region->getID()));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't load region: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(nullopt, db.getRegionCompletedStatus(region->getID()));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't get region status: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ db.deleteRegion(std::move(*region));
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't delete region: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ EXPECT_EQ(std::numeric_limits<uint64_t>::max(), db.getOfflineMapboxTileCount());
+ EXPECT_EQ(1u, log.count(warning(ResultCode::Auth, "Can't get offline Mapbox tile count: authorization denied")));
+ EXPECT_EQ(0u, log.uncheckedCount());
+
+ fs.reset();
+}
+#endif // __QT__