mirror of
https://github.com/immich-app/immich.git
synced 2025-01-21 00:52:43 -05:00
fix(mobile): validate response code from download file (#4543)
Co-authored-by: shalong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
This commit is contained in:
parent
5156d76194
commit
22172a680b
5 changed files with 97 additions and 28 deletions
|
@ -164,6 +164,7 @@
|
||||||
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
|
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
|
||||||
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
|
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
|
||||||
"image_viewer_page_state_provider_download_error": "Download Error",
|
"image_viewer_page_state_provider_download_error": "Download Error",
|
||||||
|
"image_viewer_page_state_provider_share_error": "Share Error",
|
||||||
"image_viewer_page_state_provider_download_success": "Download Success",
|
"image_viewer_page_state_provider_download_success": "Download Success",
|
||||||
"library_page_albums": "Albums",
|
"library_page_albums": "Albums",
|
||||||
"library_page_archive": "Archive",
|
"library_page_archive": "Archive",
|
||||||
|
|
|
@ -57,9 +57,19 @@ class ImageViewerStateNotifier extends StateNotifier<ImageViewerPageState> {
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext buildContext) {
|
builder: (BuildContext buildContext) {
|
||||||
_shareService
|
_shareService.shareAsset(asset).then(
|
||||||
.shareAsset(asset)
|
(bool status) {
|
||||||
.then((_) => Navigator.of(buildContext).pop());
|
if (!status) {
|
||||||
|
ImmichToast.show(
|
||||||
|
context: context,
|
||||||
|
msg: 'image_viewer_page_state_provider_share_error'.tr(),
|
||||||
|
toastType: ToastType.error,
|
||||||
|
gravity: ToastGravity.BOTTOM,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
Navigator.of(buildContext).pop();
|
||||||
|
},
|
||||||
|
);
|
||||||
return const ShareDialog();
|
return const ShareDialog();
|
||||||
},
|
},
|
||||||
barrierDismissible: false,
|
barrierDismissible: false,
|
||||||
|
|
|
@ -5,6 +5,7 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:immich_mobile/shared/models/asset.dart';
|
import 'package:immich_mobile/shared/models/asset.dart';
|
||||||
import 'package:immich_mobile/shared/providers/api.provider.dart';
|
import 'package:immich_mobile/shared/providers/api.provider.dart';
|
||||||
import 'package:immich_mobile/shared/services/api.service.dart';
|
import 'package:immich_mobile/shared/services/api.service.dart';
|
||||||
|
import 'package:logging/logging.dart';
|
||||||
|
|
||||||
import 'package:photo_manager/photo_manager.dart';
|
import 'package:photo_manager/photo_manager.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
@ -14,6 +15,7 @@ final imageViewerServiceProvider =
|
||||||
|
|
||||||
class ImageViewerService {
|
class ImageViewerService {
|
||||||
final ApiService _apiService;
|
final ApiService _apiService;
|
||||||
|
final Logger _log = Logger("ImageViewerService");
|
||||||
|
|
||||||
ImageViewerService(this._apiService);
|
ImageViewerService(this._apiService);
|
||||||
|
|
||||||
|
@ -29,6 +31,16 @@ class ImageViewerService {
|
||||||
asset.livePhotoVideoId!,
|
asset.livePhotoVideoId!,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (imageResponse.statusCode != 200 ||
|
||||||
|
motionReponse.statusCode != 200) {
|
||||||
|
final failedResponse =
|
||||||
|
imageResponse.statusCode != 200 ? imageResponse : motionReponse;
|
||||||
|
_log.severe(
|
||||||
|
"Motion asset download failed with status - ${failedResponse.statusCode} and response - ${failedResponse.body}",
|
||||||
|
);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
final AssetEntity? entity;
|
final AssetEntity? entity;
|
||||||
|
|
||||||
final tempDir = await getTemporaryDirectory();
|
final tempDir = await getTemporaryDirectory();
|
||||||
|
@ -48,6 +60,13 @@ class ImageViewerService {
|
||||||
var res = await _apiService.assetApi
|
var res = await _apiService.assetApi
|
||||||
.downloadFileWithHttpInfo(asset.remoteId!);
|
.downloadFileWithHttpInfo(asset.remoteId!);
|
||||||
|
|
||||||
|
if (res.statusCode != 200) {
|
||||||
|
_log.severe(
|
||||||
|
"Asset download failed with status - ${res.statusCode} and response - ${res.body}",
|
||||||
|
);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
final AssetEntity? entity;
|
final AssetEntity? entity;
|
||||||
|
|
||||||
if (asset.isImage) {
|
if (asset.isImage) {
|
||||||
|
|
|
@ -4,6 +4,7 @@ import 'package:flutter/material.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:immich_mobile/shared/models/asset.dart';
|
import 'package:immich_mobile/shared/models/asset.dart';
|
||||||
import 'package:immich_mobile/shared/providers/api.provider.dart';
|
import 'package:immich_mobile/shared/providers/api.provider.dart';
|
||||||
|
import 'package:logging/logging.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:share_plus/share_plus.dart';
|
import 'package:share_plus/share_plus.dart';
|
||||||
import 'api.service.dart';
|
import 'api.service.dart';
|
||||||
|
@ -13,32 +14,60 @@ final shareServiceProvider =
|
||||||
|
|
||||||
class ShareService {
|
class ShareService {
|
||||||
final ApiService _apiService;
|
final ApiService _apiService;
|
||||||
|
final Logger _log = Logger("ShareService");
|
||||||
|
|
||||||
ShareService(this._apiService);
|
ShareService(this._apiService);
|
||||||
|
|
||||||
Future<void> shareAsset(Asset asset) async {
|
Future<bool> shareAsset(Asset asset) async {
|
||||||
await shareAssets([asset]);
|
return await shareAssets([asset]);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> shareAssets(List<Asset> assets) async {
|
Future<bool> shareAssets(List<Asset> assets) async {
|
||||||
final downloadedXFiles = assets.map<Future<XFile>>((asset) async {
|
try {
|
||||||
if (asset.isRemote) {
|
final downloadedXFiles = <XFile>[];
|
||||||
final tempDir = await getTemporaryDirectory();
|
|
||||||
final fileName = asset.fileName;
|
|
||||||
final tempFile = await File('${tempDir.path}/$fileName').create();
|
|
||||||
final res = await _apiService.assetApi
|
|
||||||
.downloadFileWithHttpInfo(asset.remoteId!);
|
|
||||||
tempFile.writeAsBytesSync(res.bodyBytes);
|
|
||||||
return XFile(tempFile.path);
|
|
||||||
} else {
|
|
||||||
File? f = await asset.local!.file;
|
|
||||||
return XFile(f!.path);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
Share.shareXFiles(
|
for (var asset in assets) {
|
||||||
await Future.wait(downloadedXFiles),
|
if (asset.isRemote) {
|
||||||
sharePositionOrigin: Rect.zero,
|
final tempDir = await getTemporaryDirectory();
|
||||||
);
|
final fileName = asset.fileName;
|
||||||
|
final tempFile = await File('${tempDir.path}/$fileName').create();
|
||||||
|
final res = await _apiService.assetApi
|
||||||
|
.downloadFileWithHttpInfo(asset.remoteId!);
|
||||||
|
|
||||||
|
if (res.statusCode != 200) {
|
||||||
|
_log.severe(
|
||||||
|
"Asset download failed with status - ${res.statusCode} and response - ${res.body}",
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
tempFile.writeAsBytesSync(res.bodyBytes);
|
||||||
|
downloadedXFiles.add(XFile(tempFile.path));
|
||||||
|
} else {
|
||||||
|
File? f = await asset.local!.file;
|
||||||
|
downloadedXFiles.add(XFile(f!.path));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (downloadedXFiles.isEmpty) {
|
||||||
|
_log.warning("No asset can be retrieved for share");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (downloadedXFiles.length != assets.length) {
|
||||||
|
_log.warning(
|
||||||
|
"Partial share - Requested: ${assets.length}, Sharing: ${downloadedXFiles.length}",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Share.shareXFiles(
|
||||||
|
downloadedXFiles,
|
||||||
|
sharePositionOrigin: Rect.zero,
|
||||||
|
);
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
_log.severe("Share failed with error $error");
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:fluttertoast/fluttertoast.dart';
|
import 'package:fluttertoast/fluttertoast.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
@ -15,10 +16,19 @@ void handleShareAssets(
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext buildContext) {
|
builder: (BuildContext buildContext) {
|
||||||
ref
|
ref.watch(shareServiceProvider).shareAssets(selection.toList()).then(
|
||||||
.watch(shareServiceProvider)
|
(bool status) {
|
||||||
.shareAssets(selection.toList())
|
if (!status) {
|
||||||
.then((_) => Navigator.of(buildContext).pop());
|
ImmichToast.show(
|
||||||
|
context: context,
|
||||||
|
msg: 'image_viewer_page_state_provider_share_error'.tr(),
|
||||||
|
toastType: ToastType.error,
|
||||||
|
gravity: ToastGravity.BOTTOM,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
Navigator.of(buildContext).pop();
|
||||||
|
},
|
||||||
|
);
|
||||||
return const ShareDialog();
|
return const ShareDialog();
|
||||||
},
|
},
|
||||||
barrierDismissible: false,
|
barrierDismissible: false,
|
||||||
|
|
Loading…
Add table
Reference in a new issue