Fix(service): flatten memory usage of downloads (sacrificing download progress indication).
This commit is contained in:
		
							parent
							
								
									bb6b342d82
								
							
						
					
					
						commit
						bedd46756d
					
				| @ -358,21 +358,54 @@ class HttpFileTransferService { | |||||||
|       downloadPath = pathlib.join(tempDir.path, filename); |       downloadPath = pathlib.join(tempDir.path, filename); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     dio.Response<dynamic>? response; | 
 | ||||||
|  |     final dio.Response<dio.ResponseBody>? response; | ||||||
|     try { |     try { | ||||||
|       response = await dio.Dio().downloadUri( |       response = await dio.Dio().get<dio.ResponseBody>( | ||||||
|         Uri.parse(job.location.url), |         job.location.url, | ||||||
|         downloadPath, |         options: dio.Options( | ||||||
|         onReceiveProgress: (count, total) { |           responseType: dio.ResponseType.stream, | ||||||
|           final progress = count.toDouble() / total.toDouble(); |         ), | ||||||
|  |       ); | ||||||
|  | 
 | ||||||
|  |       final downloadStream = response.data?.stream; | ||||||
|  | 
 | ||||||
|  |       if ( downloadStream != null) { | ||||||
|  |         final totalFileSizeString = response.headers['Content-Length']?.first; | ||||||
|  |         final totalFileSize = int.parse(totalFileSizeString!); | ||||||
|  |         final file = File(downloadedPath); | ||||||
|  |         // Since acting on downloadStream events like to fire progress events | ||||||
|  |         // causes memory spikes relative to the file size, I chose to listen to | ||||||
|  |         // the created file instead. This does not cause memory spikes, but see | ||||||
|  |         // below. | ||||||
|  |         final fileSink = file.openWrite(mode: FileMode.writeOnlyAppend); | ||||||
|  |         file.watch().listen((FileSystemEvent event) { | ||||||
|  |            if (event is FileSystemCreateEvent || event is FileSystemModifyEvent) { | ||||||
|  |              final fileSize = file.lengthSync(); | ||||||
|  |             final progress = fileSize / totalFileSize; | ||||||
|             sendEvent( |             sendEvent( | ||||||
|               ProgressEvent( |               ProgressEvent( | ||||||
|                   id: job.mId, |                   id: job.mId, | ||||||
|                   progress: progress == 1 ? 0.99 : progress, |                   progress: progress == 1 ? 0.99 : progress, | ||||||
|               ), |               ), | ||||||
|             ); |             ); | ||||||
|         }, |           } | ||||||
|       ); |         }); | ||||||
|  | 
 | ||||||
|  |         // TODO(Unknown): Find some way to pause execution until the download | ||||||
|  |         // has finished. Because now, ProgressEvents are sent when the job is | ||||||
|  |         // already deleted. Below, a list of solutions I tried (for both this | ||||||
|  |         // problem and the problem I tried to solve with this workaround). | ||||||
|  | 
 | ||||||
|  |         downloadStream.listen(fileSink.add); // .onDone would cause memory spikes | ||||||
|  | 
 | ||||||
|  |         // while (! downloadStream.isEmpty) causes memory spikes | ||||||
|  |         // while (file.lengthSync() / totalFileSize != 1) causes memory spikes | ||||||
|  |         // using a controller on downloadStream causes memory spikes | ||||||
|  |         // using a controller and piping causes memory spikes | ||||||
|  |         // fileSink.addStream(downloadStream) causes memory spikes | ||||||
|  | 
 | ||||||
|  |       } | ||||||
|     } on dio.DioError catch(err) { |     } on dio.DioError catch(err) { | ||||||
|       // TODO(PapaTutuWawa): React if we received an error that is not related to the |       // TODO(PapaTutuWawa): React if we received an error that is not related to the | ||||||
|       //                     connection. |       //                     connection. | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Millesimus
						Millesimus