local_sync_service.dart 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. import 'dart:async';
  2. import 'dart:io';
  3. import 'package:computer/computer.dart';
  4. import 'package:logging/logging.dart';
  5. import 'package:photo_manager/photo_manager.dart';
  6. import 'package:photos/core/configuration.dart';
  7. import 'package:photos/core/event_bus.dart';
  8. import 'package:photos/db/files_db.dart';
  9. import 'package:photos/events/local_photos_updated_event.dart';
  10. import 'package:photos/events/sync_status_update_event.dart';
  11. import 'package:photos/models/file.dart';
  12. import 'package:photos/services/app_lifecycle_service.dart';
  13. import 'package:photos/utils/file_sync_util.dart';
  14. import 'package:shared_preferences/shared_preferences.dart';
  15. class LocalSyncService {
  16. final _logger = Logger("LocalSyncService");
  17. final _db = FilesDB.instance;
  18. final Computer _computer = Computer();
  19. SharedPreferences _prefs;
  20. Completer<void> _existingSync;
  21. static const kDbUpdationTimeKey = "db_updation_time";
  22. static const kHasCompletedFirstImportKey = "has_completed_firstImport";
  23. static const kHasGrantedPermissionsKey = "has_granted_permissions";
  24. static const kPermissionStateKey = "permission_state";
  25. static const kEditedFileIDsKey = "edited_file_ids";
  26. static const kDownloadedFileIDsKey = "downloaded_file_ids";
  27. // Adding `_2` as a suffic to pull files that were earlier ignored due to permission errors
  28. // See https://github.com/CaiJingLong/flutter_photo_manager/issues/589
  29. static const kInvalidFileIDsKey = "invalid_file_ids_2";
  30. LocalSyncService._privateConstructor();
  31. static final LocalSyncService instance = LocalSyncService._privateConstructor();
  32. Future<void> init() async {
  33. _prefs = await SharedPreferences.getInstance();
  34. if (!AppLifecycleService.instance.isForeground) {
  35. await PhotoManager.setIgnorePermissionCheck(true);
  36. }
  37. await _computer.turnOn(workersCount: 1);
  38. if (hasGrantedPermissions()) {
  39. _registerChangeCallback();
  40. }
  41. }
  42. Future<void> sync() async {
  43. if (!_prefs.containsKey(kHasGrantedPermissionsKey)) {
  44. _logger.info("Skipping local sync since permission has not been granted");
  45. return;
  46. }
  47. if (Platform.isAndroid && AppLifecycleService.instance.isForeground) {
  48. final permissionState = await PhotoManager.requestPermissionExtend();
  49. if (permissionState != PermissionState.authorized) {
  50. _logger.severe(
  51. "sync requested with invalid permission",
  52. permissionState.toString(),
  53. );
  54. return;
  55. }
  56. }
  57. if (_existingSync != null) {
  58. _logger.warning("Sync already in progress, skipping.");
  59. return _existingSync.future;
  60. }
  61. _existingSync = Completer<void>();
  62. final existingLocalFileIDs = await _db.getExistingLocalFileIDs();
  63. _logger.info(
  64. existingLocalFileIDs.length.toString() + " localIDs were discovered",
  65. );
  66. final editedFileIDs = getEditedFileIDs().toSet();
  67. final downloadedFileIDs = getDownloadedFileIDs().toSet();
  68. final syncStartTime = DateTime.now().microsecondsSinceEpoch;
  69. final lastDBUpdationTime = _prefs.getInt(kDbUpdationTimeKey) ?? 0;
  70. final startTime = DateTime.now().microsecondsSinceEpoch;
  71. if (lastDBUpdationTime != 0) {
  72. await _loadAndStorePhotos(
  73. lastDBUpdationTime,
  74. syncStartTime,
  75. existingLocalFileIDs,
  76. editedFileIDs,
  77. downloadedFileIDs,
  78. );
  79. } else {
  80. // Load from 0 - 01.01.2010
  81. Bus.instance.fire(SyncStatusUpdate(SyncStatus.startedFirstGalleryImport));
  82. var startTime = 0;
  83. var toYear = 2010;
  84. var toTime = DateTime(toYear).microsecondsSinceEpoch;
  85. while (toTime < syncStartTime) {
  86. await _loadAndStorePhotos(
  87. startTime,
  88. toTime,
  89. existingLocalFileIDs,
  90. editedFileIDs,
  91. downloadedFileIDs,
  92. );
  93. startTime = toTime;
  94. toYear++;
  95. toTime = DateTime(toYear).microsecondsSinceEpoch;
  96. }
  97. await _loadAndStorePhotos(
  98. startTime,
  99. syncStartTime,
  100. existingLocalFileIDs,
  101. editedFileIDs,
  102. downloadedFileIDs,
  103. );
  104. }
  105. if (!_prefs.containsKey(kHasCompletedFirstImportKey) ||
  106. !_prefs.getBool(kHasCompletedFirstImportKey)) {
  107. await _prefs.setBool(kHasCompletedFirstImportKey, true);
  108. _logger.fine("first gallery import finished");
  109. Bus.instance.fire(SyncStatusUpdate(SyncStatus.completedFirstGalleryImport));
  110. }
  111. final endTime = DateTime.now().microsecondsSinceEpoch;
  112. final duration = Duration(microseconds: endTime - startTime);
  113. _logger.info("Load took " + duration.inMilliseconds.toString() + "ms");
  114. _existingSync.complete();
  115. _existingSync = null;
  116. }
  117. Future<bool> syncAll() async {
  118. final sTime = DateTime.now().microsecondsSinceEpoch;
  119. final localAssets = await getAllLocalAssets();
  120. final eTime = DateTime.now().microsecondsSinceEpoch;
  121. final d = Duration(microseconds: eTime - sTime);
  122. _logger.info(
  123. "Loading from the beginning returned " +
  124. localAssets.length.toString() +
  125. " assets and took " +
  126. d.inMilliseconds.toString() +
  127. "ms",
  128. );
  129. final existingIDs = await _db.getExistingLocalFileIDs();
  130. final invalidIDs = getInvalidFileIDs().toSet();
  131. final unsyncedFiles = await getUnsyncedFiles(localAssets, existingIDs, invalidIDs, _computer);
  132. if (unsyncedFiles.isNotEmpty) {
  133. await _db.insertMultiple(unsyncedFiles);
  134. _logger.info(
  135. "Inserted " + unsyncedFiles.length.toString() + " unsynced files.",
  136. );
  137. _updatePathsToBackup(unsyncedFiles);
  138. Bus.instance.fire(LocalPhotosUpdatedEvent(unsyncedFiles));
  139. return true;
  140. }
  141. return false;
  142. }
  143. Future<void> trackEditedFile(File file) async {
  144. final editedIDs = getEditedFileIDs();
  145. editedIDs.add(file.localID);
  146. await _prefs.setStringList(kEditedFileIDsKey, editedIDs);
  147. }
  148. List<String> getEditedFileIDs() {
  149. if (_prefs.containsKey(kEditedFileIDsKey)) {
  150. return _prefs.getStringList(kEditedFileIDsKey);
  151. } else {
  152. List<String> editedIDs = [];
  153. return editedIDs;
  154. }
  155. }
  156. Future<void> trackDownloadedFile(String localID) async {
  157. final downloadedIDs = getDownloadedFileIDs();
  158. downloadedIDs.add(localID);
  159. await _prefs.setStringList(kDownloadedFileIDsKey, downloadedIDs);
  160. }
  161. List<String> getDownloadedFileIDs() {
  162. if (_prefs.containsKey(kDownloadedFileIDsKey)) {
  163. return _prefs.getStringList(kDownloadedFileIDsKey);
  164. } else {
  165. List<String> downloadedIDs = [];
  166. return downloadedIDs;
  167. }
  168. }
  169. Future<void> trackInvalidFile(File file) async {
  170. final invalidIDs = getInvalidFileIDs();
  171. invalidIDs.add(file.localID);
  172. await _prefs.setStringList(kInvalidFileIDsKey, invalidIDs);
  173. }
  174. List<String> getInvalidFileIDs() {
  175. if (_prefs.containsKey(kInvalidFileIDsKey)) {
  176. return _prefs.getStringList(kInvalidFileIDsKey);
  177. } else {
  178. List<String> invalidIDs = [];
  179. return invalidIDs;
  180. }
  181. }
  182. bool hasGrantedPermissions() {
  183. return _prefs.getBool(kHasGrantedPermissionsKey) ?? false;
  184. }
  185. bool hasGrantedLimitedPermissions() {
  186. return _prefs.getString(kPermissionStateKey) == PermissionState.limited.toString();
  187. }
  188. Future<void> onPermissionGranted(PermissionState state) async {
  189. await _prefs.setBool(kHasGrantedPermissionsKey, true);
  190. await _prefs.setString(kPermissionStateKey, state.toString());
  191. _registerChangeCallback();
  192. }
  193. bool hasCompletedFirstImport() {
  194. return _prefs.getBool(kHasCompletedFirstImportKey) ?? false;
  195. }
  196. Future<void> _loadAndStorePhotos(
  197. int fromTime,
  198. int toTime,
  199. Set<String> existingLocalFileIDs,
  200. Set<String> editedFileIDs,
  201. Set<String> downloadedFileIDs,
  202. ) async {
  203. _logger.info(
  204. "Loading photos from " +
  205. DateTime.fromMicrosecondsSinceEpoch(fromTime).toString() +
  206. " to " +
  207. DateTime.fromMicrosecondsSinceEpoch(toTime).toString(),
  208. );
  209. final files = await getDeviceFiles(fromTime, toTime, _computer);
  210. if (files.isNotEmpty) {
  211. _logger.info("Fetched " + files.length.toString() + " files.");
  212. final updatedFiles =
  213. files.where((file) => existingLocalFileIDs.contains(file.localID)).toList();
  214. updatedFiles.removeWhere((file) => editedFileIDs.contains(file.localID));
  215. updatedFiles.removeWhere((file) => downloadedFileIDs.contains(file.localID));
  216. if (updatedFiles.isNotEmpty) {
  217. _logger.info(
  218. updatedFiles.length.toString() + " local files were updated.",
  219. );
  220. }
  221. for (final file in updatedFiles) {
  222. await captureUpdateLogs(file);
  223. await _db.updateUploadedFile(
  224. file.localID,
  225. file.title,
  226. file.location,
  227. file.creationTime,
  228. file.modificationTime,
  229. null,
  230. );
  231. }
  232. final List<File> allFiles = [];
  233. allFiles.addAll(files);
  234. files.removeWhere((file) => existingLocalFileIDs.contains(file.localID));
  235. await _db.insertMultiple(files);
  236. _logger.info("Inserted " + files.length.toString() + " files.");
  237. _updatePathsToBackup(files);
  238. Bus.instance.fire(LocalPhotosUpdatedEvent(allFiles));
  239. }
  240. await _prefs.setInt(kDbUpdationTimeKey, toTime);
  241. }
  242. // _captureUpdateLogs is a helper method to log details
  243. // about the file which is being marked for re-upload
  244. Future<void> captureUpdateLogs(File file) async {
  245. _logger.info(
  246. 're-upload locally updated file ${file.toString()}',
  247. );
  248. try {
  249. if (Platform.isIOS) {
  250. var assetEntity = await AssetEntity.fromId(file.localID);
  251. if (assetEntity != null) {
  252. var isLocallyAvailable = await assetEntity.isLocallyAvailable(isOrigin: true);
  253. _logger.info(
  254. 're-upload asset ${file.toString()} with localAvailableFlag '
  255. '$isLocallyAvailable and fav ${assetEntity.isFavorite}',
  256. );
  257. } else {
  258. _logger.info('re-upload failed to fetch assetInfo ${file.toString()}');
  259. }
  260. }
  261. } catch (ignore) {
  262. //ignore
  263. }
  264. }
  265. void _updatePathsToBackup(List<File> files) {
  266. if (Configuration.instance.hasSelectedAllFoldersForBackup()) {
  267. final pathsToBackup = Configuration.instance.getPathsToBackUp();
  268. final newFilePaths = files.map((file) => file.deviceFolder).toList();
  269. pathsToBackup.addAll(newFilePaths);
  270. Configuration.instance.setPathsToBackUp(pathsToBackup);
  271. }
  272. }
  273. void _registerChangeCallback() {
  274. // In case of iOS limit permission, this call back is fired immediately
  275. // after file selection dialog is dismissed.
  276. PhotoManager.addChangeCallback((value) async {
  277. _logger.info("Something changed on disk");
  278. if (_existingSync != null) {
  279. await _existingSync.future;
  280. }
  281. if (hasGrantedLimitedPermissions()) {
  282. syncAll();
  283. } else {
  284. sync();
  285. }
  286. });
  287. PhotoManager.startChangeNotify();
  288. }
  289. }