summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKonstantin Käfer <mail@kkaefer.com>2016-05-27 13:37:41 +0200
committerKonstantin Käfer <mail@kkaefer.com>2016-06-10 12:42:14 +0200
commit91895b9aecedc9c080e18b2e7304f5d2a8da340a (patch)
tree7ffce27fdd12aa2092f07873603e07e48ea489ca
parentf180179d1b98e6610263b71ea0d9ef7f44725005 (diff)
downloadqtlocation-mapboxgl-91895b9aecedc9c080e18b2e7304f5d2a8da340a.tar.gz
[core] only perform optional requests if the FileSource supports it
-rw-r--r--include/mbgl/storage/default_file_source.hpp4
-rw-r--r--include/mbgl/storage/file_source.hpp8
-rw-r--r--src/mbgl/tile/file_based_tile_source.hpp1
-rw-r--r--src/mbgl/tile/file_based_tile_source_impl.hpp51
4 files changed, 50 insertions, 14 deletions
diff --git a/include/mbgl/storage/default_file_source.hpp b/include/mbgl/storage/default_file_source.hpp
index e77befda1b..235e738254 100644
--- a/include/mbgl/storage/default_file_source.hpp
+++ b/include/mbgl/storage/default_file_source.hpp
@@ -26,6 +26,10 @@ public:
uint64_t maximumCacheSize = util::DEFAULT_MAX_CACHE_SIZE);
~DefaultFileSource() override;
+ bool supportsOptionalRequests() const override {
+ return true;
+ }
+
void setAccessToken(const std::string&);
std::string getAccessToken() const;
diff --git a/include/mbgl/storage/file_source.hpp b/include/mbgl/storage/file_source.hpp
index 88e0336187..404c683fdb 100644
--- a/include/mbgl/storage/file_source.hpp
+++ b/include/mbgl/storage/file_source.hpp
@@ -23,6 +23,14 @@ public:
// If the request is cancelled before the callback is executed, the callback will
// not be executed.
virtual std::unique_ptr<AsyncRequest> request(const Resource&, Callback) = 0;
+
+ // When a file source supports optional requests, it must return true.
+ // Optional requests are requests that aren't as urgent, but could be useful, e.g.
+ // to cover part of the map while loading. The FileSource should only do cheap actions to
+ // retrieve the data, e.g. load it from a cache, but not from the internet.
+ virtual bool supportsOptionalRequests() const {
+ return false;
+ }
};
} // namespace mbgl
diff --git a/src/mbgl/tile/file_based_tile_source.hpp b/src/mbgl/tile/file_based_tile_source.hpp
index 72d0782710..df4dbbfccc 100644
--- a/src/mbgl/tile/file_based_tile_source.hpp
+++ b/src/mbgl/tile/file_based_tile_source.hpp
@@ -19,6 +19,7 @@ protected:
void makeOptional() override;
private:
+ void loadOptional();
void loadedData(const Response&);
void loadRequired();
diff --git a/src/mbgl/tile/file_based_tile_source_impl.hpp b/src/mbgl/tile/file_based_tile_source_impl.hpp
index c64789ca3c..4a11728332 100644
--- a/src/mbgl/tile/file_based_tile_source_impl.hpp
+++ b/src/mbgl/tile/file_based_tile_source_impl.hpp
@@ -13,12 +13,42 @@ FileBasedTileSource<T, I>::FileBasedTileSource(typename T::data_type& tileData_,
FileSource& fileSource_)
: T(tileData_), resource(resource_), fileSource(fileSource_) {
assert(!request);
- // The first request is always optional.
+ if (fileSource.supportsOptionalRequests()) {
+ // When supported, the first request is always optional, even if the TileSource
+ // is marked as required. That way, we can let the first optional request continue
+ // to load when the TileSource is later changed from required to optional. If we
+ // started out with a required request, we'd have to cancel everything, including the
+ // initial optional part of the request.
+ loadOptional();
+ } else {
+ // When the FileSource doesn't support optional requests, we do nothing until the
+ // data is definitely required.
+ if (T::isRequired()) {
+ loadRequired();
+ } else {
+ // We're using this field to check whether the pending request is optional or required.
+ resource.necessity = Resource::Optional;
+ }
+ }
+}
+
+template <typename T, typename I>
+void FileBasedTileSource<T, I>::loadOptional() {
+ assert(!request);
+
resource.necessity = Resource::Optional;
request = fileSource.request(resource, [this](Response res) {
request.reset();
- loadedData(res);
- T::loaded = true;
+
+ if (res.error && res.error->reason == Response::Error::Reason::NotFound) {
+ // When the optional request could not be satisfied, don't treat it as an error.
+ // Instead, we make sure that the next request knows that there has been an optional
+ // request before by setting one of the prior* fields.
+ resource.priorExpires = Timestamp{ Seconds::zero() };
+ } else {
+ loadedData(res);
+ }
+
if (T::isRequired()) {
loadRequired();
}
@@ -27,14 +57,14 @@ FileBasedTileSource<T, I>::FileBasedTileSource(typename T::data_type& tileData_,
template <typename T, typename I>
void FileBasedTileSource<T, I>::makeRequired() {
- if (T::loaded && !request) {
+ if (!request) {
loadRequired();
}
}
template <typename T, typename I>
void FileBasedTileSource<T, I>::makeOptional() {
- if (T::loaded && request) {
+ if (resource.necessity == Resource::Required && request) {
// Abort a potential HTTP request.
request.reset();
}
@@ -42,12 +72,7 @@ void FileBasedTileSource<T, I>::makeOptional() {
template <typename T, typename I>
void FileBasedTileSource<T, I>::loadedData(const Response& res) {
- if (res.error && !T::loaded && res.error->reason == Response::Error::Reason::NotFound) {
- // When the optional request could not be satisfied, don't treat it as an error. Instead,
- // we make sure that the next request knows that there has been an optional request before
- // by setting one of the prior* fields.
- resource.priorExpires = Timestamp{ Seconds::zero() };
- } else if (res.error) {
+ if (res.error && res.error->reason != Response::Error::Reason::NotFound) {
T::tileData.setError(std::make_exception_ptr(std::runtime_error(res.error->message)));
} else if (res.notModified) {
resource.priorExpires = res.expires;
@@ -67,9 +92,7 @@ void FileBasedTileSource<T, I>::loadRequired() {
assert(!request);
resource.necessity = Resource::Required;
- request = fileSource.request(resource, [this](Response res) {
- loadedData(res);
- });
+ request = fileSource.request(resource, [this](Response res) { loadedData(res); });
}
} // namespace mbgl