From 4a937450e6a089549e9977fefd6b0e56351f13d5 Mon Sep 17 00:00:00 2001 From: Gabriel Araujo Date: Fri, 14 Jul 2023 00:03:57 -0300 Subject: [PATCH] =?UTF-8?q?=E2=9C=A8=20Make=20`MultipartFile`=20clonable?= =?UTF-8?q?=20(#1889)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Add restoration capability to MultipartFile - Add tests to ensure MultipartFile is not changed during cloning, it just gets a new instance that can be consumed by FormData Possible next steps: Create method in FormData to make it recoverable/clonable as well. ### New Pull Request Checklist - [x] I have read the [Documentation](https://pub.dev/documentation/dio/latest/) - [x] I have searched for a similar pull request in the [project](https://github.com/cfug/dio/pulls) and found none - [x] I have updated this branch with the latest `main` branch to avoid conflicts (via merge from master or rebase) - [x] I have added the required tests to prove the fix/feature I'm adding - [x] I have updated the documentation (if necessary) - [x] I have run the tests without failures - [x] I have updated the `CHANGELOG.md` in the corresponding package --------- Signed-off-by: Gabriel Araujo Signed-off-by: Peter Leibiger Co-authored-by: Peter Leibiger --- dio/CHANGELOG.md | 3 +- dio/lib/src/multipart_file.dart | 11 ++++ dio/test/formdata_test.dart | 93 +++++++++++++++++++++++++++++++++ 3 files changed, 106 insertions(+), 1 deletion(-) diff --git a/dio/CHANGELOG.md b/dio/CHANGELOG.md index bf791d6fe..ceb2726d9 100644 --- a/dio/CHANGELOG.md +++ b/dio/CHANGELOG.md @@ -6,6 +6,7 @@ See the [Migration Guide][] for the complete breaking changes list.** ## Unreleased - Remove `http` from `dev_dependencies`. +- Add support for cloning `MultipartFile` from `FormData`. - Only produce null response body when `ResponseType.json`. ## 5.2.1+1 @@ -29,7 +30,7 @@ See the [Migration Guide][] for the complete breaking changes list.** Dio 6.0.0 - Please use the replacement `IOHttpClientAdapter.createHttpClient` instead. - Using `CancelToken` no longer closes and re-creates `HttpClient` for each request when `IOHttpClientAdapter` is used. - Fix timeout handling for browser `receiveTimeout`. -- Improve performance when sending binary data (`List`/`Uint8List`). +- Improve performance when sending binary data (`List`/`Uint8List`). ## 5.1.2 diff --git a/dio/lib/src/multipart_file.dart b/dio/lib/src/multipart_file.dart index 85fa3e6c0..6bcb42f26 100644 --- a/dio/lib/src/multipart_file.dart +++ b/dio/lib/src/multipart_file.dart @@ -140,4 +140,15 @@ class MultipartFile { _isFinalized = true; return _stream; } + + /// Clone MultipartFile, returning a new instance of the same object. + /// This is useful if your request failed and you wish to retry it, + /// such as an unauthorized exception can be solved by refreshing the token. + MultipartFile clone() => MultipartFile( + _stream, + length, + filename: filename, + contentType: contentType, + headers: headers, + ); } diff --git a/dio/test/formdata_test.dart b/dio/test/formdata_test.dart index b1171ab4b..c1ae7b216 100644 --- a/dio/test/formdata_test.dart +++ b/dio/test/formdata_test.dart @@ -94,6 +94,99 @@ void main() async { testOn: 'vm', ); + // Cloned multipart files should be able to be read again and be the same + // as the original ones. + test( + 'complex with cloning', + () async { + final multipartFile1 = MultipartFile.fromString( + 'hello world.', + headers: { + 'test': ['a'] + }, + ); + final multipartFile2 = await MultipartFile.fromFile( + 'test/mock/_testfile', + filename: '1.txt', + headers: { + 'test': ['b'] + }, + ); + final multipartFile3 = MultipartFile.fromFileSync( + 'test/mock/_testfile', + filename: '2.txt', + headers: { + 'test': ['c'] + }, + ); + + final fm = FormData.fromMap({ + 'name': 'wendux', + 'age': 25, + 'path': '/图片空间/地址', + 'file': multipartFile1, + 'files': [ + multipartFile2, + multipartFile3, + ] + }); + final fmStr = await fm.readAsBytes(); + + // Files are finalized after being read. + try { + multipartFile1.finalize(); + fail('Should not be able to finalize a file twice.'); + } catch (e) { + expect(e, isA()); + expect( + (e as StateError).message, + 'The MultipartFile has already been finalized. This typically ' + 'means you are using the same MultipartFile in repeated requests.', + ); + } + + final fm1 = FormData(); + fm1.fields.add(MapEntry('name', 'wendux')); + fm1.fields.add(MapEntry('age', '25')); + fm1.fields.add(MapEntry('path', '/图片空间/地址')); + fm1.files.add( + MapEntry( + 'file', + multipartFile1.clone(), + ), + ); + fm1.files.add( + MapEntry( + 'files', + multipartFile2.clone(), + ), + ); + fm1.files.add( + MapEntry( + 'files', + multipartFile3.clone(), + ), + ); + expect(fmStr.length, fm1.length); + + // The cloned multipart files should be able to be read again. + expect(fm.files[0].value.isFinalized, true); + expect(fm.files[1].value.isFinalized, true); + expect(fm.files[2].value.isFinalized, true); + expect(fm1.files[0].value.isFinalized, false); + expect(fm1.files[1].value.isFinalized, false); + expect(fm1.files[2].value.isFinalized, false); + + // The cloned multipart files' properties should be the same as the + // original ones. + expect(fm1.files[0].value.filename, multipartFile1.filename); + expect(fm1.files[0].value.contentType, multipartFile1.contentType); + expect(fm1.files[0].value.length, multipartFile1.length); + expect(fm1.files[0].value.headers, multipartFile1.headers); + }, + testOn: 'vm', + ); + test('encodes maps correctly', () async { final fd = FormData.fromMap( {