Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add timeout parameter for downloading #482

Open
wants to merge 2 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 15 additions & 4 deletions flutter_cache_manager/lib/src/cache_manager.dart
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,8 @@ class CacheManager implements BaseCacheManager {

/// Get the file from the cache and/or online, depending on availability and age.
/// Downloaded form [url], [headers] can be used for example for authentication.
/// [timeout] can be used to specify a timeout for the download, and it will throw
/// a [TimeoutException] when the download takes longer than the specified timeout.
/// The files are returned as stream. First the cached file if available, when the
/// cached file is too old the newly downloaded file is returned afterwards.
///
Expand All @@ -108,11 +110,17 @@ class CacheManager implements BaseCacheManager {
/// returned from the cache there will be no progress given, although the file
/// might be outdated and a new file is being downloaded in the background.
@override
Stream<FileResponse> getFileStream(String url,
{String? key, Map<String, String>? headers, bool withProgress = false}) {
Stream<FileResponse> getFileStream(
String url, {
String? key,
Map<String, String>? headers,
Duration? timeout,
bool withProgress = false,
}) {
key ??= url;
final streamController = StreamController<FileResponse>();
_pushFileToStream(streamController, url, key, headers, withProgress);
_pushFileToStream(
streamController, url, key, headers, timeout, withProgress);
return streamController.stream;
}

Expand All @@ -121,6 +129,7 @@ class CacheManager implements BaseCacheManager {
String url,
String? key,
Map<String, String>? headers,
Duration? timeout,
bool withProgress,
) async {
key ??= url;
Expand All @@ -139,7 +148,7 @@ class CacheManager implements BaseCacheManager {
if (cacheFile == null || cacheFile.validTill.isBefore(DateTime.now())) {
try {
await for (final response
in _webHelper.downloadFile(url, key: key, authHeaders: headers)) {
in _webHelper.downloadFile(url, key: key, authHeaders: headers, timeout: timeout)) {
if (response is DownloadProgress && withProgress) {
streamController.add(response);
}
Expand Down Expand Up @@ -173,13 +182,15 @@ class CacheManager implements BaseCacheManager {
Future<FileInfo> downloadFile(String url,
{String? key,
Map<String, String>? authHeaders,
Duration? timeout,
bool force = false}) async {
key ??= url;
final fileResponse = await _webHelper
.downloadFile(
url,
key: key,
authHeaders: authHeaders,
timeout: timeout,
ignoreMemCache: force,
)
.firstWhere((r) => r is FileInfo);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ abstract class BaseCacheManager {

/// Get the file from the cache and/or online, depending on availability and age.
/// Downloaded form [url], [headers] can be used for example for authentication.
/// [timeout] can be used to specify a timeout for the download, and it will throw
/// a [TimeoutException] when the download takes longer than the specified timeout.
/// The files are returned as stream. First the cached file if available, when the
/// cached file is too old the newly downloaded file is returned afterwards.
///
Expand All @@ -38,11 +40,17 @@ abstract class BaseCacheManager {
/// returned from the cache there will be no progress given, although the file
/// might be outdated and a new file is being downloaded in the background.
Stream<FileResponse> getFileStream(String url,
{String? key, Map<String, String>? headers, bool withProgress});
{String? key,
Map<String, String>? headers,
Duration? timeout,
bool withProgress});

///Download the file and add to cache
Future<FileInfo> downloadFile(String url,
{String? key, Map<String, String>? authHeaders, bool force = false});
{String? key,
Map<String, String>? authHeaders,
Duration? timeout,
bool force = false});

/// Get the file from the cache.
/// Specify [ignoreMemCache] to force a re-read from the database
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,19 @@ mixin ImageCacheManager on BaseCacheManager {
String url, {
String? key,
Map<String, String>? headers,
Duration? timeout,
bool withProgress = false,
int? maxHeight,
int? maxWidth,
}) async* {
if (maxHeight == null && maxWidth == null) {
yield* getFileStream(url,
key: key, headers: headers, withProgress: withProgress);
yield* getFileStream(
url,
key: key,
headers: headers,
timeout: timeout,
withProgress: withProgress,
);
return;
}
key ??= url;
Expand Down
9 changes: 6 additions & 3 deletions flutter_cache_manager/lib/src/compat/file_service_compat.dart
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,12 @@ class FileServiceCompat extends FileService {

@override
Future<FileServiceResponse> get(String url,
{Map<String, String>? headers}) async {
final legacyResponse = await fileFetcher(url, headers: headers);
return CompatFileServiceGetResponse(legacyResponse);
{Map<String, String>? headers, Duration? timeout}) async {
var legacyResponse = fileFetcher(url, headers: headers);
if (timeout != null) {
legacyResponse = legacyResponse.timeout(timeout);
}
return CompatFileServiceGetResponse(await legacyResponse);
}
}

Expand Down
16 changes: 12 additions & 4 deletions flutter_cache_manager/lib/src/web/file_service.dart
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,11 @@ import 'package:http/http.dart' as http;
abstract class FileService {
int concurrentFetches = 10;

Future<FileServiceResponse> get(String url, {Map<String, String>? headers});
Future<FileServiceResponse> get(
String url, {
Map<String, String>? headers,
Duration? timeout,
});
}

/// [HttpFileService] is the most common file service and the default for
Expand All @@ -29,14 +33,18 @@ class HttpFileService extends FileService {

@override
Future<FileServiceResponse> get(String url,
{Map<String, String>? headers}) async {
{Map<String, String>? headers, Duration? timeout}) async {
final req = http.Request('GET', Uri.parse(url));
if (headers != null) {
req.headers.addAll(headers);
}
final httpResponse = await _httpClient.send(req);

return HttpGetResponse(httpResponse);
var streamedResponse = _httpClient.send(req);
if (timeout != null) {
streamedResponse = streamedResponse.timeout(timeout);
}

return HttpGetResponse(await streamedResponse);
}
}

Expand Down
25 changes: 15 additions & 10 deletions flutter_cache_manager/lib/src/web/web_helper.dart
Original file line number Diff line number Diff line change
Expand Up @@ -32,13 +32,14 @@ class WebHelper {
Stream<FileResponse> downloadFile(String url,
{String? key,
Map<String, String>? authHeaders,
Duration? timeout,
bool ignoreMemCache = false}) {
key ??= url;
var subject = _memCache[key];
if (subject == null || ignoreMemCache) {
subject = BehaviorSubject<FileResponse>();
_memCache[key] = subject;
_downloadOrAddToQueue(url, key, authHeaders);
_downloadOrAddToQueue(url, key, authHeaders, timeout);
}
return subject.stream;
}
Expand All @@ -49,6 +50,7 @@ class WebHelper {
String url,
String key,
Map<String, String>? authHeaders,
Duration? timeout,
) async {
//Add to queue if there are too many calls.
if (concurrentCalls >= fileFetcher.concurrentFetches) {
Expand All @@ -61,8 +63,8 @@ class WebHelper {
concurrentCalls++;
final subject = _memCache[key]!;
try {
await for (final result
in _updateFile(url, key, authHeaders: authHeaders)) {
await for (final result in _updateFile(url, key,
authHeaders: authHeaders, timeout: timeout)) {
subject.add(result);
}
} on Object catch (e, stackTrace) {
Expand All @@ -71,19 +73,19 @@ class WebHelper {
concurrentCalls--;
await subject.close();
_memCache.remove(key);
_checkQueue();
_checkQueue(timeout);
}
}

void _checkQueue() {
void _checkQueue(Duration? timeout) {
if (_queue.isEmpty) return;
final next = _queue.removeFirst();
_downloadOrAddToQueue(next.url, next.key, next.headers);
_downloadOrAddToQueue(next.url, next.key, next.headers, timeout);
}

///Download the file from the url
Stream<FileResponse> _updateFile(String url, String key,
{Map<String, String>? authHeaders}) async* {
{Map<String, String>? authHeaders, Duration? timeout}) async* {
var cacheObject = await _store.retrieveCacheData(key);
cacheObject = cacheObject == null
? CacheObject(
Expand All @@ -93,12 +95,15 @@ class WebHelper {
relativePath: '${const Uuid().v1()}.file',
)
: cacheObject.copyWith(url: url);
final response = await _download(cacheObject, authHeaders);
final response = await _download(cacheObject, authHeaders, timeout);
yield* _manageResponse(cacheObject, response);
}

Future<FileServiceResponse> _download(
CacheObject cacheObject, Map<String, String>? authHeaders) {
CacheObject cacheObject,
Map<String, String>? authHeaders,
Duration? timeout,
) {
final headers = <String, String>{};

final etag = cacheObject.eTag;
Expand All @@ -112,7 +117,7 @@ class WebHelper {
headers.addAll(authHeaders);
}

return fileFetcher.get(cacheObject.url, headers: headers);
return fileFetcher.get(cacheObject.url, headers: headers, timeout: timeout);
}

Stream<FileResponse> _manageResponse(
Expand Down
13 changes: 11 additions & 2 deletions flutter_cache_manager/test/mock.mocks.dart
Original file line number Diff line number Diff line change
Expand Up @@ -443,20 +443,27 @@ class MockFileServiceBase extends _i1.Mock implements _i3.FileService {
_i4.Future<_i3.FileServiceResponse> get(
String? url, {
Map<String, String>? headers,
Duration? timeout,
}) =>
(super.noSuchMethod(
Invocation.method(
#get,
[url],
{#headers: headers},
{
#headers: headers,
#timeout: timeout,
},
),
returnValue: _i4.Future<_i3.FileServiceResponse>.value(
_FakeFileServiceResponse_4(
this,
Invocation.method(
#get,
[url],
{#headers: headers},
{
#headers: headers,
#timeout: timeout,
},
),
)),
) as _i4.Future<_i3.FileServiceResponse>);
Expand Down Expand Up @@ -499,6 +506,7 @@ class MockWebHelper extends _i1.Mock implements _i7.WebHelper {
String? url, {
String? key,
Map<String, String>? authHeaders,
Duration? timeout,
bool? ignoreMemCache = false,
}) =>
(super.noSuchMethod(
Expand All @@ -508,6 +516,7 @@ class MockWebHelper extends _i1.Mock implements _i7.WebHelper {
{
#key: key,
#authHeaders: authHeaders,
#timeout: timeout,
#ignoreMemCache: ignoreMemCache,
},
),
Expand Down