Unverified Commit 839fdbd2 authored by Alexander Aprelev's avatar Alexander Aprelev Committed by GitHub

Retry devfs uploads in case they fail. (#41406)

* Retry devfs uploads in case they fail.

Fixes #34959.
parent 69af9ade
...@@ -289,7 +289,7 @@ class _DevFSHttpWriter { ...@@ -289,7 +289,7 @@ class _DevFSHttpWriter {
while ((_inFlight < kMaxInFlight) && (!_completer.isCompleted) && _outstanding.isNotEmpty) { while ((_inFlight < kMaxInFlight) && (!_completer.isCompleted) && _outstanding.isNotEmpty) {
final Uri deviceUri = _outstanding.keys.first; final Uri deviceUri = _outstanding.keys.first;
final DevFSContent content = _outstanding.remove(deviceUri); final DevFSContent content = _outstanding.remove(deviceUri);
_startWrite(deviceUri, content); _startWrite(deviceUri, content, retry: 10);
_inFlight += 1; _inFlight += 1;
} }
if ((_inFlight == 0) && (!_completer.isCompleted) && _outstanding.isEmpty) { if ((_inFlight == 0) && (!_completer.isCompleted) && _outstanding.isEmpty) {
...@@ -299,9 +299,10 @@ class _DevFSHttpWriter { ...@@ -299,9 +299,10 @@ class _DevFSHttpWriter {
Future<void> _startWrite( Future<void> _startWrite(
Uri deviceUri, Uri deviceUri,
DevFSContent content, [ DevFSContent content, {
int retry = 0, int retry = 0,
]) async { }) async {
while(true) {
try { try {
final HttpClientRequest request = await _client.putUrl(httpAddress); final HttpClientRequest request = await _client.putUrl(httpAddress);
request.headers.removeAll(HttpHeaders.acceptEncodingHeader); request.headers.removeAll(HttpHeaders.acceptEncodingHeader);
...@@ -310,13 +311,23 @@ class _DevFSHttpWriter { ...@@ -310,13 +311,23 @@ class _DevFSHttpWriter {
final Stream<List<int>> contents = content.contentsAsCompressedStream(); final Stream<List<int>> contents = content.contentsAsCompressedStream();
await request.addStream(contents); await request.addStream(contents);
final HttpClientResponse response = await request.close(); final HttpClientResponse response = await request.close();
await response.drain<void>(); response.listen((_) => null,
onError: (dynamic error) { printTrace('error: $error'); },
cancelOnError: true);
break;
} catch (error, trace) { } catch (error, trace) {
if (!_completer.isCompleted) { if (!_completer.isCompleted) {
printTrace('Error writing "$deviceUri" to DevFS: $error'); printTrace('Error writing "$deviceUri" to DevFS: $error');
if (retry > 0) {
retry--;
printTrace('trying again in a few - $retry more attempts left');
await Future<void>.delayed(const Duration(milliseconds: 500));
continue;
}
_completer.completeError(error, trace); _completer.completeError(error, trace);
} }
} }
}
_inFlight -= 1; _inFlight -= 1;
_scheduleWrites(); _scheduleWrites();
} }
......
...@@ -92,6 +92,75 @@ void main() { ...@@ -92,6 +92,75 @@ void main() {
}, skip: Platform.isWindows); // TODO(jonahwilliams): fix or disable this functionality. }, skip: Platform.isWindows); // TODO(jonahwilliams): fix or disable this functionality.
}); });
group('mocked http client', () {
HttpOverrides savedHttpOverrides;
HttpClient httpClient;
setUpAll(() {
tempDir = _newTempDir(fs);
basePath = tempDir.path;
savedHttpOverrides = HttpOverrides.current;
httpClient = MockOddlyFailingHttpClient();
HttpOverrides.global = MyHttpOverrides(httpClient);
});
tearDownAll(() async {
HttpOverrides.global = savedHttpOverrides;
});
testUsingContext('retry uploads when failure', () async {
final File file = fs.file(fs.path.join(basePath, filePath));
await file.parent.create(recursive: true);
file.writeAsBytesSync(<int>[1, 2, 3]);
// simulate package
await _createPackage(fs, 'somepkg', 'somefile.txt');
final RealMockVMService vmService = RealMockVMService();
final RealMockVM vm = RealMockVM();
final Map<String, dynamic> response = <String, dynamic>{ 'uri': 'file://abc' };
when(vm.createDevFS(any)).thenAnswer((Invocation invocation) {
return Future<Map<String, dynamic>>.value(response);
});
when(vmService.vm).thenReturn(vm);
reset(httpClient);
final MockHttpClientRequest httpRequest = MockHttpClientRequest();
when(httpRequest.headers).thenReturn(MockHttpHeaders());
when(httpClient.putUrl(any)).thenAnswer((Invocation invocation) {
return Future<HttpClientRequest>.value(httpRequest);
});
final MockHttpClientResponse httpClientResponse = MockHttpClientResponse();
int nRequest = 0;
const int kFailedAttempts = 5;
when(httpRequest.close()).thenAnswer((Invocation invocation) {
if (nRequest++ < kFailedAttempts) {
throw 'Connection resert by peer';
}
return Future<HttpClientResponse>.value(httpClientResponse);
});
devFS = DevFS(vmService, 'test', tempDir);
await devFS.create();
final MockResidentCompiler residentCompiler = MockResidentCompiler();
final UpdateFSReport report = await devFS.update(
mainPath: 'lib/foo.txt',
generator: residentCompiler,
pathToReload: 'lib/foo.txt.dill',
trackWidgetCreation: false,
invalidatedFiles: <Uri>[],
);
expect(report.syncedBytes, 22);
expect(report.success, isTrue);
verify(httpClient.putUrl(any)).called(kFailedAttempts + 1);
verify(httpRequest.close()).called(kFailedAttempts + 1);
}, overrides: <Type, Generator>{
FileSystem: () => fs,
});
});
group('devfs remote', () { group('devfs remote', () {
MockVMService vmService; MockVMService vmService;
final MockResidentCompiler residentCompiler = MockResidentCompiler(); final MockResidentCompiler residentCompiler = MockResidentCompiler();
...@@ -200,7 +269,6 @@ void main() { ...@@ -200,7 +269,6 @@ void main() {
}, overrides: <Type, Generator>{ }, overrides: <Type, Generator>{
FileSystem: () => fs, FileSystem: () => fs,
}); });
}); });
} }
...@@ -326,3 +394,25 @@ Future<void> _createPackage(FileSystem fs, String pkgName, String pkgFileName, { ...@@ -326,3 +394,25 @@ Future<void> _createPackage(FileSystem fs, String pkgName, String pkgFileName, {
fs.file(fs.path.join(_tempDirs[0].path, '.packages')).writeAsStringSync(sb.toString()); fs.file(fs.path.join(_tempDirs[0].path, '.packages')).writeAsStringSync(sb.toString());
} }
class RealMockVM extends Mock implements VM {
}
class RealMockVMService extends Mock implements VMService {
}
class MyHttpOverrides extends HttpOverrides {
MyHttpOverrides(this._httpClient);
@override
HttpClient createHttpClient(SecurityContext context) {
return _httpClient;
}
final HttpClient _httpClient;
}
class MockOddlyFailingHttpClient extends Mock implements HttpClient {}
class MockHttpClientRequest extends Mock implements HttpClientRequest {}
class MockHttpHeaders extends Mock implements HttpHeaders {}
class MockHttpClientResponse extends Mock implements HttpClientResponse {}
\ No newline at end of file
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment