files_db.dart 41 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301
  1. import 'dart:io';
  2. import 'package:logging/logging.dart';
  3. import 'package:path/path.dart';
  4. import 'package:path_provider/path_provider.dart';
  5. import 'package:photos/models/backup_status.dart';
  6. import 'package:photos/models/file.dart';
  7. import 'package:photos/models/file_load_result.dart';
  8. import 'package:photos/models/file_type.dart';
  9. import 'package:photos/models/location.dart';
  10. import 'package:photos/models/magic_metadata.dart';
  11. import 'package:photos/utils/file_uploader_util.dart';
  12. import 'package:sqflite/sqflite.dart';
  13. import 'package:sqflite_migration/sqflite_migration.dart';
  14. class FilesDB {
  15. /*
  16. Note: columnUploadedFileID and columnCollectionID have to be compared against
  17. both NULL and -1 because older clients might have entries where the DEFAULT
  18. was unset, and a migration script to set the DEFAULT would break in case of
  19. duplicate entries for un-uploaded files that were created due to a collision
  20. in background and foreground syncs.
  21. */
  22. static const _databaseName = "ente.files.db";
  23. static final Logger _logger = Logger("FilesDB");
  24. static const table = 'files';
  25. static const tempTable = 'temp_files';
  26. static const columnGeneratedID = '_id';
  27. static const columnUploadedFileID = 'uploaded_file_id';
  28. static const columnOwnerID = 'owner_id';
  29. static const columnCollectionID = 'collection_id';
  30. static const columnLocalID = 'local_id';
  31. static const columnTitle = 'title';
  32. static const columnDeviceFolder = 'device_folder';
  33. static const columnLatitude = 'latitude';
  34. static const columnLongitude = 'longitude';
  35. static const columnFileType = 'file_type';
  36. static const columnFileSubType = 'file_sub_type';
  37. static const columnDuration = 'duration';
  38. static const columnExif = 'exif';
  39. static const columnHash = 'hash';
  40. static const columnMetadataVersion = 'metadata_version';
  41. static const columnIsDeleted = 'is_deleted';
  42. static const columnCreationTime = 'creation_time';
  43. static const columnModificationTime = 'modification_time';
  44. static const columnUpdationTime = 'updation_time';
  45. static const columnEncryptedKey = 'encrypted_key';
  46. static const columnKeyDecryptionNonce = 'key_decryption_nonce';
  47. static const columnFileDecryptionHeader = 'file_decryption_header';
  48. static const columnThumbnailDecryptionHeader = 'thumbnail_decryption_header';
  49. static const columnMetadataDecryptionHeader = 'metadata_decryption_header';
  50. // MMD -> Magic Metadata
  51. static const columnMMdEncodedJson = 'mmd_encoded_json';
  52. static const columnMMdVersion = 'mmd_ver';
  53. static const columnPubMMdEncodedJson = 'pub_mmd_encoded_json';
  54. static const columnPubMMdVersion = 'pub_mmd_ver';
  55. // part of magic metadata
  56. // Only parse & store selected fields from JSON in separate columns if
  57. // we need to write query based on that field
  58. static const columnMMdVisibility = 'mmd_visibility';
  59. static final initializationScript = [...createTable(table)];
  60. static final migrationScripts = [
  61. ...alterDeviceFolderToAllowNULL(),
  62. ...alterTimestampColumnTypes(),
  63. ...addIndices(),
  64. ...addMetadataColumns(),
  65. ...addMagicMetadataColumns(),
  66. ...addUniqueConstraintOnCollectionFiles(),
  67. ...addPubMagicMetadataColumns()
  68. ];
  69. final dbConfig = MigrationConfig(
  70. initializationScript: initializationScript,
  71. migrationScripts: migrationScripts,
  72. );
  73. // make this a singleton class
  74. FilesDB._privateConstructor();
  75. static final FilesDB instance = FilesDB._privateConstructor();
  76. // only have a single app-wide reference to the database
  77. static Future<Database> _dbFuture;
  78. Future<Database> get database async {
  79. // lazily instantiate the db the first time it is accessed
  80. _dbFuture ??= _initDatabase();
  81. return _dbFuture;
  82. }
  83. // this opens the database (and creates it if it doesn't exist)
  84. Future<Database> _initDatabase() async {
  85. final Directory documentsDirectory = await getApplicationDocumentsDirectory();
  86. final String path = join(documentsDirectory.path, _databaseName);
  87. _logger.info("DB path " + path);
  88. return await openDatabaseWithMigration(path, dbConfig);
  89. }
  90. // SQL code to create the database table
  91. static List<String> createTable(String tableName) {
  92. return [
  93. '''
  94. CREATE TABLE $tableName (
  95. $columnGeneratedID INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
  96. $columnLocalID TEXT,
  97. $columnUploadedFileID INTEGER DEFAULT -1,
  98. $columnOwnerID INTEGER,
  99. $columnCollectionID INTEGER DEFAULT -1,
  100. $columnTitle TEXT NOT NULL,
  101. $columnDeviceFolder TEXT,
  102. $columnLatitude REAL,
  103. $columnLongitude REAL,
  104. $columnFileType INTEGER,
  105. $columnModificationTime TEXT NOT NULL,
  106. $columnEncryptedKey TEXT,
  107. $columnKeyDecryptionNonce TEXT,
  108. $columnFileDecryptionHeader TEXT,
  109. $columnThumbnailDecryptionHeader TEXT,
  110. $columnMetadataDecryptionHeader TEXT,
  111. $columnIsDeleted INTEGER DEFAULT 0,
  112. $columnCreationTime TEXT NOT NULL,
  113. $columnUpdationTime TEXT,
  114. UNIQUE($columnLocalID, $columnUploadedFileID, $columnCollectionID)
  115. );
  116. ''',
  117. ];
  118. }
  119. static List<String> addIndices() {
  120. return [
  121. '''
  122. CREATE INDEX IF NOT EXISTS collection_id_index ON $table($columnCollectionID);
  123. ''',
  124. '''
  125. CREATE INDEX IF NOT EXISTS device_folder_index ON $table($columnDeviceFolder);
  126. ''',
  127. '''
  128. CREATE INDEX IF NOT EXISTS creation_time_index ON $table($columnCreationTime);
  129. ''',
  130. '''
  131. CREATE INDEX IF NOT EXISTS updation_time_index ON $table($columnUpdationTime);
  132. '''
  133. ];
  134. }
  135. static List<String> alterDeviceFolderToAllowNULL() {
  136. return [
  137. ...createTable(tempTable),
  138. '''
  139. INSERT INTO $tempTable
  140. SELECT *
  141. FROM $table;
  142. DROP TABLE $table;
  143. ALTER TABLE $tempTable
  144. RENAME TO $table;
  145. '''
  146. ];
  147. }
  148. static List<String> alterTimestampColumnTypes() {
  149. return [
  150. '''
  151. DROP TABLE IF EXISTS $tempTable;
  152. ''',
  153. '''
  154. CREATE TABLE $tempTable (
  155. $columnGeneratedID INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
  156. $columnLocalID TEXT,
  157. $columnUploadedFileID INTEGER DEFAULT -1,
  158. $columnOwnerID INTEGER,
  159. $columnCollectionID INTEGER DEFAULT -1,
  160. $columnTitle TEXT NOT NULL,
  161. $columnDeviceFolder TEXT,
  162. $columnLatitude REAL,
  163. $columnLongitude REAL,
  164. $columnFileType INTEGER,
  165. $columnModificationTime INTEGER NOT NULL,
  166. $columnEncryptedKey TEXT,
  167. $columnKeyDecryptionNonce TEXT,
  168. $columnFileDecryptionHeader TEXT,
  169. $columnThumbnailDecryptionHeader TEXT,
  170. $columnMetadataDecryptionHeader TEXT,
  171. $columnCreationTime INTEGER NOT NULL,
  172. $columnUpdationTime INTEGER,
  173. UNIQUE($columnLocalID, $columnUploadedFileID, $columnCollectionID)
  174. );
  175. ''',
  176. '''
  177. INSERT INTO $tempTable
  178. SELECT
  179. $columnGeneratedID,
  180. $columnLocalID,
  181. $columnUploadedFileID,
  182. $columnOwnerID,
  183. $columnCollectionID,
  184. $columnTitle,
  185. $columnDeviceFolder,
  186. $columnLatitude,
  187. $columnLongitude,
  188. $columnFileType,
  189. CAST($columnModificationTime AS INTEGER),
  190. $columnEncryptedKey,
  191. $columnKeyDecryptionNonce,
  192. $columnFileDecryptionHeader,
  193. $columnThumbnailDecryptionHeader,
  194. $columnMetadataDecryptionHeader,
  195. CAST($columnCreationTime AS INTEGER),
  196. CAST($columnUpdationTime AS INTEGER)
  197. FROM $table;
  198. ''',
  199. '''
  200. DROP TABLE $table;
  201. ''',
  202. '''
  203. ALTER TABLE $tempTable
  204. RENAME TO $table;
  205. ''',
  206. ];
  207. }
  208. static List<String> addMetadataColumns() {
  209. return [
  210. '''
  211. ALTER TABLE $table ADD COLUMN $columnFileSubType INTEGER;
  212. ''',
  213. '''
  214. ALTER TABLE $table ADD COLUMN $columnDuration INTEGER;
  215. ''',
  216. '''
  217. ALTER TABLE $table ADD COLUMN $columnExif TEXT;
  218. ''',
  219. '''
  220. ALTER TABLE $table ADD COLUMN $columnHash TEXT;
  221. ''',
  222. '''
  223. ALTER TABLE $table ADD COLUMN $columnMetadataVersion INTEGER;
  224. ''',
  225. ];
  226. }
  227. static List<String> addMagicMetadataColumns() {
  228. return [
  229. '''
  230. ALTER TABLE $table ADD COLUMN $columnMMdEncodedJson TEXT DEFAULT '{}';
  231. ''',
  232. '''
  233. ALTER TABLE $table ADD COLUMN $columnMMdVersion INTEGER DEFAULT 0;
  234. ''',
  235. '''
  236. ALTER TABLE $table ADD COLUMN $columnMMdVisibility INTEGER DEFAULT $kVisibilityVisible;
  237. '''
  238. ];
  239. }
  240. static List<String> addUniqueConstraintOnCollectionFiles() {
  241. return [
  242. '''
  243. DELETE from $table where $columnCollectionID || '-' || $columnUploadedFileID IN
  244. (SELECT $columnCollectionID || '-' || $columnUploadedFileID from $table WHERE
  245. $columnCollectionID is not NULL AND $columnUploadedFileID is NOT NULL
  246. AND $columnCollectionID != -1 AND $columnUploadedFileID != -1
  247. GROUP BY ($columnCollectionID || '-' || $columnUploadedFileID) HAVING count(*) > 1)
  248. AND ($columnCollectionID || '-' || $columnUploadedFileID || '-' || $columnGeneratedID) NOT IN
  249. (SELECT $columnCollectionID || '-' || $columnUploadedFileID || '-' || max($columnGeneratedID)
  250. from $table WHERE
  251. $columnCollectionID is not NULL AND $columnUploadedFileID is NOT NULL
  252. AND $columnCollectionID != -1 AND $columnUploadedFileID != -1 GROUP BY
  253. ($columnCollectionID || '-' || $columnUploadedFileID) HAVING count(*) > 1);
  254. ''',
  255. '''
  256. CREATE UNIQUE INDEX IF NOT EXISTS cid_uid ON $table ($columnCollectionID, $columnUploadedFileID)
  257. WHERE $columnCollectionID is not NULL AND $columnUploadedFileID is not NULL
  258. AND $columnCollectionID != -1 AND $columnUploadedFileID != -1;
  259. '''
  260. ];
  261. }
  262. static List<String> addPubMagicMetadataColumns() {
  263. return [
  264. '''
  265. ALTER TABLE $table ADD COLUMN $columnPubMMdEncodedJson TEXT DEFAULT '{}';
  266. ''',
  267. '''
  268. ALTER TABLE $table ADD COLUMN $columnPubMMdVersion INTEGER DEFAULT 0;
  269. '''
  270. ];
  271. }
  272. Future<void> clearTable() async {
  273. final db = await instance.database;
  274. await db.delete(table);
  275. }
  276. Future<void> insertMultiple(List<File> files) async {
  277. final startTime = DateTime.now();
  278. final db = await instance.database;
  279. var batch = db.batch();
  280. int batchCounter = 0;
  281. for (File file in files) {
  282. if (batchCounter == 400) {
  283. await batch.commit(noResult: true);
  284. batch = db.batch();
  285. batchCounter = 0;
  286. }
  287. batch.insert(
  288. table,
  289. _getRowForFile(file),
  290. conflictAlgorithm: ConflictAlgorithm.replace,
  291. );
  292. batchCounter++;
  293. }
  294. await batch.commit(noResult: true);
  295. final endTime = DateTime.now();
  296. final duration = Duration(
  297. microseconds:
  298. endTime.microsecondsSinceEpoch - startTime.microsecondsSinceEpoch,
  299. );
  300. _logger.info(
  301. "Batch insert of " +
  302. files.length.toString() +
  303. " took " +
  304. duration.inMilliseconds.toString() +
  305. "ms.",
  306. );
  307. }
  308. Future<int> insert(File file) async {
  309. final db = await instance.database;
  310. return db.insert(
  311. table,
  312. _getRowForFile(file),
  313. conflictAlgorithm: ConflictAlgorithm.replace,
  314. );
  315. }
  316. Future<File> getFile(int generatedID) async {
  317. final db = await instance.database;
  318. final results = await db.query(
  319. table,
  320. where: '$columnGeneratedID = ?',
  321. whereArgs: [generatedID],
  322. );
  323. if (results.isEmpty) {
  324. return null;
  325. }
  326. return _convertToFiles(results)[0];
  327. }
  328. Future<File> getUploadedFile(int uploadedID, int collectionID) async {
  329. final db = await instance.database;
  330. final results = await db.query(
  331. table,
  332. where: '$columnUploadedFileID = ? AND $columnCollectionID = ?',
  333. whereArgs: [
  334. uploadedID,
  335. collectionID,
  336. ],
  337. );
  338. if (results.isEmpty) {
  339. return null;
  340. }
  341. return _convertToFiles(results)[0];
  342. }
  343. Future<Set<int>> getUploadedFileIDs(int collectionID) async {
  344. final db = await instance.database;
  345. final results = await db.query(
  346. table,
  347. columns: [columnUploadedFileID],
  348. where: '$columnCollectionID = ?',
  349. whereArgs: [
  350. collectionID,
  351. ],
  352. );
  353. final ids = <int>{};
  354. for (final result in results) {
  355. ids.add(result[columnUploadedFileID]);
  356. }
  357. return ids;
  358. }
  359. Future<BackedUpFileIDs> getBackedUpIDs() async {
  360. final db = await instance.database;
  361. final results = await db.query(
  362. table,
  363. columns: [columnLocalID, columnUploadedFileID],
  364. where:
  365. '$columnLocalID IS NOT NULL AND ($columnUploadedFileID IS NOT NULL AND $columnUploadedFileID IS NOT -1)',
  366. );
  367. final localIDs = <String>{};
  368. final uploadedIDs = <int>{};
  369. for (final result in results) {
  370. localIDs.add(result[columnLocalID]);
  371. uploadedIDs.add(result[columnUploadedFileID]);
  372. }
  373. return BackedUpFileIDs(localIDs.toList(), uploadedIDs.toList());
  374. }
  375. Future<FileLoadResult> getAllUploadedFiles(
  376. int startTime,
  377. int endTime,
  378. int ownerID, {
  379. int limit,
  380. bool asc,
  381. int visibility = kVisibilityVisible,
  382. Set<int> ignoredCollectionIDs,
  383. }) async {
  384. final db = await instance.database;
  385. final order = (asc ?? false ? 'ASC' : 'DESC');
  386. final results = await db.query(
  387. table,
  388. where:
  389. '$columnCreationTime >= ? AND $columnCreationTime <= ? AND $columnOwnerID = ? AND ($columnCollectionID IS NOT NULL AND $columnCollectionID IS NOT -1)'
  390. ' AND $columnMMdVisibility = ?',
  391. whereArgs: [startTime, endTime, ownerID, visibility],
  392. orderBy:
  393. '$columnCreationTime ' + order + ', $columnModificationTime ' + order,
  394. limit: limit,
  395. );
  396. final files = _convertToFiles(results);
  397. final List<File> deduplicatedFiles =
  398. _deduplicatedAndFilterIgnoredFiles(files, ignoredCollectionIDs);
  399. return FileLoadResult(deduplicatedFiles, files.length == limit);
  400. }
  401. Future<FileLoadResult> getAllLocalAndUploadedFiles(
  402. int startTime,
  403. int endTime,
  404. int ownerID, {
  405. int limit,
  406. bool asc,
  407. Set<int> ignoredCollectionIDs,
  408. }) async {
  409. final db = await instance.database;
  410. final order = (asc ?? false ? 'ASC' : 'DESC');
  411. final results = await db.query(
  412. table,
  413. where:
  414. '$columnCreationTime >= ? AND $columnCreationTime <= ? AND ($columnOwnerID IS NULL OR $columnOwnerID = ?) AND ($columnMMdVisibility IS NULL OR $columnMMdVisibility = ?)'
  415. ' AND ($columnLocalID IS NOT NULL OR ($columnCollectionID IS NOT NULL AND $columnCollectionID IS NOT -1))',
  416. whereArgs: [startTime, endTime, ownerID, kVisibilityVisible],
  417. orderBy:
  418. '$columnCreationTime ' + order + ', $columnModificationTime ' + order,
  419. limit: limit,
  420. );
  421. final files = _convertToFiles(results);
  422. final List<File> deduplicatedFiles =
  423. _deduplicatedAndFilterIgnoredFiles(files, ignoredCollectionIDs);
  424. return FileLoadResult(deduplicatedFiles, files.length == limit);
  425. }
  426. Future<FileLoadResult> getImportantFiles(
  427. int startTime,
  428. int endTime,
  429. int ownerID,
  430. List<String> paths, {
  431. int limit,
  432. bool asc,
  433. Set<int> ignoredCollectionIDs,
  434. }) async {
  435. final db = await instance.database;
  436. String inParam = "";
  437. for (final path in paths) {
  438. inParam += "'" + path.replaceAll("'", "''") + "',";
  439. }
  440. inParam = inParam.substring(0, inParam.length - 1);
  441. final order = (asc ?? false ? 'ASC' : 'DESC');
  442. final results = await db.query(
  443. table,
  444. where:
  445. '$columnCreationTime >= ? AND $columnCreationTime <= ? AND ($columnOwnerID IS NULL OR $columnOwnerID = ?) AND ($columnMMdVisibility IS NULL OR $columnMMdVisibility = ?)'
  446. 'AND (($columnLocalID IS NOT NULL AND $columnDeviceFolder IN ($inParam)) OR ($columnCollectionID IS NOT NULL AND $columnCollectionID IS NOT -1))',
  447. whereArgs: [startTime, endTime, ownerID, kVisibilityVisible],
  448. orderBy:
  449. '$columnCreationTime ' + order + ', $columnModificationTime ' + order,
  450. limit: limit,
  451. );
  452. final files = _convertToFiles(results);
  453. final List<File> deduplicatedFiles =
  454. _deduplicatedAndFilterIgnoredFiles(files, ignoredCollectionIDs);
  455. return FileLoadResult(deduplicatedFiles, files.length == limit);
  456. }
  457. List<File> _deduplicateByLocalID(List<File> files) {
  458. final localIDs = <String>{};
  459. final List<File> deduplicatedFiles = [];
  460. for (final file in files) {
  461. final id = file.localID;
  462. if (id != null && localIDs.contains(id)) {
  463. continue;
  464. }
  465. localIDs.add(id);
  466. deduplicatedFiles.add(file);
  467. }
  468. return deduplicatedFiles;
  469. }
  470. List<File> _deduplicatedAndFilterIgnoredFiles(
  471. List<File> files,
  472. Set<int> ignoredCollectionIDs,
  473. ) {
  474. final uploadedFileIDs = <int>{};
  475. final List<File> deduplicatedFiles = [];
  476. for (final file in files) {
  477. final id = file.uploadedFileID;
  478. if (ignoredCollectionIDs != null &&
  479. ignoredCollectionIDs.contains(file.collectionID)) {
  480. continue;
  481. }
  482. if (id != null && id != -1 && uploadedFileIDs.contains(id)) {
  483. continue;
  484. }
  485. uploadedFileIDs.add(id);
  486. deduplicatedFiles.add(file);
  487. }
  488. return deduplicatedFiles;
  489. }
  490. Future<FileLoadResult> getFilesInCollection(
  491. int collectionID,
  492. int startTime,
  493. int endTime, {
  494. int limit,
  495. bool asc,
  496. }) async {
  497. final db = await instance.database;
  498. final order = (asc ?? false ? 'ASC' : 'DESC');
  499. final results = await db.query(
  500. table,
  501. where:
  502. '$columnCollectionID = ? AND $columnCreationTime >= ? AND $columnCreationTime <= ?',
  503. whereArgs: [collectionID, startTime, endTime],
  504. orderBy:
  505. '$columnCreationTime ' + order + ', $columnModificationTime ' + order,
  506. limit: limit,
  507. );
  508. final files = _convertToFiles(results);
  509. _logger.info("Fetched " + files.length.toString() + " files");
  510. return FileLoadResult(files, files.length == limit);
  511. }
  512. Future<FileLoadResult> getFilesInPath(
  513. String path,
  514. int startTime,
  515. int endTime, {
  516. int limit,
  517. bool asc,
  518. }) async {
  519. final db = await instance.database;
  520. final order = (asc ?? false ? 'ASC' : 'DESC');
  521. final results = await db.query(
  522. table,
  523. where:
  524. '$columnDeviceFolder = ? AND $columnCreationTime >= ? AND $columnCreationTime <= ? AND $columnLocalID IS NOT NULL',
  525. whereArgs: [path, startTime, endTime],
  526. orderBy:
  527. '$columnCreationTime ' + order + ', $columnModificationTime ' + order,
  528. groupBy: columnLocalID,
  529. limit: limit,
  530. );
  531. final files = _convertToFiles(results);
  532. return FileLoadResult(files, files.length == limit);
  533. }
  534. Future<FileLoadResult> getLocalDeviceFiles(
  535. int startTime,
  536. int endTime, {
  537. int limit,
  538. bool asc,
  539. }) async {
  540. final db = await instance.database;
  541. final order = (asc ?? false ? 'ASC' : 'DESC');
  542. final results = await db.query(
  543. table,
  544. where:
  545. '$columnCreationTime >= ? AND $columnCreationTime <= ? AND $columnLocalID IS NOT NULL',
  546. whereArgs: [startTime, endTime],
  547. orderBy:
  548. '$columnCreationTime ' + order + ', $columnModificationTime ' + order,
  549. limit: limit,
  550. );
  551. final files = _convertToFiles(results);
  552. final result = _deduplicateByLocalID(files);
  553. return FileLoadResult(result, files.length == limit);
  554. }
  555. Future<List<File>> getAllVideos() async {
  556. final db = await instance.database;
  557. final results = await db.query(
  558. table,
  559. where: '$columnLocalID IS NOT NULL AND $columnFileType = 1',
  560. orderBy: '$columnCreationTime DESC',
  561. );
  562. return _convertToFiles(results);
  563. }
  564. Future<List<File>> getAllInPath(String path) async {
  565. final db = await instance.database;
  566. final results = await db.query(
  567. table,
  568. where: '$columnLocalID IS NOT NULL AND $columnDeviceFolder = ?',
  569. whereArgs: [path],
  570. orderBy: '$columnCreationTime DESC',
  571. groupBy: columnLocalID,
  572. );
  573. return _convertToFiles(results);
  574. }
  575. Future<List<File>> getFilesCreatedWithinDurations(
  576. List<List<int>> durations,
  577. Set<int> ignoredCollectionIDs, {
  578. String order = 'ASC',
  579. }) async {
  580. final db = await instance.database;
  581. String whereClause = "( ";
  582. for (int index = 0; index < durations.length; index++) {
  583. whereClause += "($columnCreationTime > " +
  584. durations[index][0].toString() +
  585. " AND $columnCreationTime < " +
  586. durations[index][1].toString() +
  587. ")";
  588. if (index != durations.length - 1) {
  589. whereClause += " OR ";
  590. }
  591. }
  592. whereClause += ") AND $columnMMdVisibility = $kVisibilityVisible";
  593. final results = await db.query(
  594. table,
  595. where: whereClause,
  596. orderBy: '$columnCreationTime ' + order,
  597. );
  598. final files = _convertToFiles(results);
  599. return _deduplicatedAndFilterIgnoredFiles(files, ignoredCollectionIDs);
  600. }
  601. Future<List<File>> getFilesToBeUploadedWithinFolders(
  602. Set<String> folders,
  603. ) async {
  604. if (folders.isEmpty) {
  605. return [];
  606. }
  607. final db = await instance.database;
  608. String inParam = "";
  609. for (final folder in folders) {
  610. inParam += "'" + folder.replaceAll("'", "''") + "',";
  611. }
  612. inParam = inParam.substring(0, inParam.length - 1);
  613. final results = await db.query(
  614. table,
  615. where:
  616. '($columnUploadedFileID IS NULL OR $columnUploadedFileID IS -1) AND $columnDeviceFolder IN ($inParam)',
  617. orderBy: '$columnCreationTime DESC',
  618. groupBy: columnLocalID,
  619. );
  620. return _convertToFiles(results);
  621. }
  622. // Files which user added to a collection manually but they are not uploaded yet.
  623. Future<List<File>> getPendingManualUploads() async {
  624. final db = await instance.database;
  625. final results = await db.query(
  626. table,
  627. where:
  628. '($columnUploadedFileID IS NULL OR $columnUploadedFileID IS -1) AND '
  629. '$columnCollectionID IS NOT NULL AND $columnCollectionID IS NOT -1 AND '
  630. '$columnLocalID IS NOT NULL AND $columnLocalID IS NOT -1',
  631. orderBy: '$columnCreationTime DESC',
  632. groupBy: columnLocalID,
  633. );
  634. final files = _convertToFiles(results);
  635. // future-safe filter just to ensure that the query doesn't end up returning files
  636. // which should not be backed up
  637. files.removeWhere(
  638. (e) =>
  639. e.collectionID == null ||
  640. e.localID == null ||
  641. e.uploadedFileID != null,
  642. );
  643. return files;
  644. }
  645. Future<List<File>> getAllLocalFiles() async {
  646. final db = await instance.database;
  647. final results = await db.query(
  648. table,
  649. where:
  650. '($columnUploadedFileID IS NULL OR $columnUploadedFileID IS -1) AND $columnLocalID IS NOT NULL',
  651. orderBy: '$columnCreationTime DESC',
  652. groupBy: columnLocalID,
  653. );
  654. return _convertToFiles(results);
  655. }
  656. Future<List<File>> getEditedRemoteFiles() async {
  657. final db = await instance.database;
  658. final results = await db.query(
  659. table,
  660. where:
  661. '($columnCollectionID IS NOT NULL AND $columnCollectionID IS NOT -1) AND ($columnUploadedFileID IS NULL OR $columnUploadedFileID IS -1)',
  662. orderBy: '$columnCreationTime DESC',
  663. groupBy: columnLocalID,
  664. );
  665. return _convertToFiles(results);
  666. }
  667. Future<List<int>> getUploadedFileIDsToBeUpdated() async {
  668. final db = await instance.database;
  669. final rows = await db.query(
  670. table,
  671. columns: [columnUploadedFileID],
  672. where:
  673. '($columnLocalID IS NOT NULL AND ($columnUploadedFileID IS NOT NULL AND $columnUploadedFileID IS NOT -1) AND $columnUpdationTime IS NULL)',
  674. orderBy: '$columnCreationTime DESC',
  675. distinct: true,
  676. );
  677. final uploadedFileIDs = <int>[];
  678. for (final row in rows) {
  679. uploadedFileIDs.add(row[columnUploadedFileID]);
  680. }
  681. return uploadedFileIDs;
  682. }
  683. Future<File> getUploadedFileInAnyCollection(int uploadedFileID) async {
  684. final db = await instance.database;
  685. final results = await db.query(
  686. table,
  687. where: '$columnUploadedFileID = ?',
  688. whereArgs: [
  689. uploadedFileID,
  690. ],
  691. limit: 1,
  692. );
  693. if (results.isEmpty) {
  694. return null;
  695. }
  696. return _convertToFiles(results)[0];
  697. }
  698. Future<Set<String>> getExistingLocalFileIDs() async {
  699. final db = await instance.database;
  700. final rows = await db.query(
  701. table,
  702. columns: [columnLocalID],
  703. distinct: true,
  704. where: '$columnLocalID IS NOT NULL',
  705. );
  706. final result = <String>{};
  707. for (final row in rows) {
  708. result.add(row[columnLocalID]);
  709. }
  710. return result;
  711. }
  712. Future<int> getNumberOfUploadedFiles() async {
  713. final db = await instance.database;
  714. final rows = await db.query(
  715. table,
  716. columns: [columnUploadedFileID],
  717. where:
  718. '($columnLocalID IS NOT NULL AND ($columnUploadedFileID IS NOT NULL AND $columnUploadedFileID IS NOT -1) AND $columnUpdationTime IS NOT NULL)',
  719. distinct: true,
  720. );
  721. return rows.length;
  722. }
  723. Future<int> updateUploadedFile(
  724. String localID,
  725. String title,
  726. Location location,
  727. int creationTime,
  728. int modificationTime,
  729. int updationTime,
  730. ) async {
  731. final db = await instance.database;
  732. return await db.update(
  733. table,
  734. {
  735. columnTitle: title,
  736. columnLatitude: location.latitude,
  737. columnLongitude: location.longitude,
  738. columnCreationTime: creationTime,
  739. columnModificationTime: modificationTime,
  740. columnUpdationTime: updationTime,
  741. },
  742. where: '$columnLocalID = ?',
  743. whereArgs: [localID],
  744. );
  745. }
  746. Future<List<File>> getMatchingFiles(
  747. String title,
  748. String deviceFolder,
  749. ) async {
  750. final db = await instance.database;
  751. final rows = await db.query(
  752. table,
  753. where: '''$columnTitle=? AND $columnDeviceFolder=?''',
  754. whereArgs: [
  755. title,
  756. deviceFolder,
  757. ],
  758. );
  759. if (rows.isNotEmpty) {
  760. return _convertToFiles(rows);
  761. } else {
  762. return null;
  763. }
  764. }
  765. Future<List<File>> getUploadedFilesWithHashes(
  766. FileHashData hashData,
  767. FileType fileType,
  768. int ownerID,
  769. ) async {
  770. String inParam = "'${hashData.fileHash}'";
  771. if (fileType == FileType.livePhoto && hashData.zipHash != null) {
  772. inParam += ",'${hashData.zipHash}'";
  773. }
  774. final db = await instance.database;
  775. final rows = await db.query(
  776. table,
  777. where: '($columnUploadedFileID != NULL OR $columnUploadedFileID != -1) '
  778. 'AND $columnOwnerID = ? AND $columnFileType ='
  779. ' ? '
  780. 'AND $columnHash IN ($inParam)',
  781. whereArgs: [
  782. ownerID,
  783. getInt(fileType),
  784. ],
  785. );
  786. return _convertToFiles(rows);
  787. }
  788. Future<int> update(File file) async {
  789. final db = await instance.database;
  790. return await db.update(
  791. table,
  792. _getRowForFile(file),
  793. where: '$columnGeneratedID = ?',
  794. whereArgs: [file.generatedID],
  795. );
  796. }
  797. Future<int> updateUploadedFileAcrossCollections(File file) async {
  798. final db = await instance.database;
  799. return await db.update(
  800. table,
  801. _getRowForFileWithoutCollection(file),
  802. where: '$columnUploadedFileID = ?',
  803. whereArgs: [file.uploadedFileID],
  804. );
  805. }
  806. Future<int> delete(int uploadedFileID) async {
  807. final db = await instance.database;
  808. return db.delete(
  809. table,
  810. where: '$columnUploadedFileID =?',
  811. whereArgs: [uploadedFileID],
  812. );
  813. }
  814. Future<int> deleteByGeneratedID(int genID) async {
  815. final db = await instance.database;
  816. return db.delete(
  817. table,
  818. where: '$columnGeneratedID =?',
  819. whereArgs: [genID],
  820. );
  821. }
  822. Future<int> deleteMultipleUploadedFiles(List<int> uploadedFileIDs) async {
  823. final db = await instance.database;
  824. return await db.delete(
  825. table,
  826. where: '$columnUploadedFileID IN (${uploadedFileIDs.join(', ')})',
  827. );
  828. }
  829. Future<int> deleteLocalFile(File file) async {
  830. final db = await instance.database;
  831. if (file.localID != null) {
  832. // delete all files with same local ID
  833. return db.delete(
  834. table,
  835. where: '$columnLocalID =?',
  836. whereArgs: [file.localID],
  837. );
  838. } else {
  839. return db.delete(
  840. table,
  841. where: '$columnGeneratedID =?',
  842. whereArgs: [file.generatedID],
  843. );
  844. }
  845. }
  846. Future<void> deleteLocalFiles(List<String> localIDs) async {
  847. String inParam = "";
  848. for (final localID in localIDs) {
  849. inParam += "'" + localID + "',";
  850. }
  851. inParam = inParam.substring(0, inParam.length - 1);
  852. final db = await instance.database;
  853. await db.rawQuery(
  854. '''
  855. UPDATE $table
  856. SET $columnLocalID = NULL
  857. WHERE $columnLocalID IN ($inParam);
  858. ''',
  859. );
  860. }
  861. Future<List<File>> getLocalFiles(List<String> localIDs) async {
  862. String inParam = "";
  863. for (final localID in localIDs) {
  864. inParam += "'" + localID + "',";
  865. }
  866. inParam = inParam.substring(0, inParam.length - 1);
  867. final db = await instance.database;
  868. final results = await db.query(
  869. table,
  870. where: '$columnLocalID IN ($inParam)',
  871. );
  872. return _convertToFiles(results);
  873. }
  874. Future<int> deleteUnSyncedLocalFiles(List<String> localIDs) async {
  875. String inParam = "";
  876. for (final localID in localIDs) {
  877. inParam += "'" + localID + "',";
  878. }
  879. inParam = inParam.substring(0, inParam.length - 1);
  880. final db = await instance.database;
  881. return db.delete(
  882. table,
  883. where:
  884. '($columnUploadedFileID is NULL OR $columnUploadedFileID = -1 ) AND $columnLocalID IN ($inParam)',
  885. );
  886. }
  887. Future<int> deleteFromCollection(int uploadedFileID, int collectionID) async {
  888. final db = await instance.database;
  889. return db.delete(
  890. table,
  891. where: '$columnUploadedFileID = ? AND $columnCollectionID = ?',
  892. whereArgs: [uploadedFileID, collectionID],
  893. );
  894. }
  895. Future<int> deleteFilesFromCollection(
  896. int collectionID,
  897. List<int> uploadedFileIDs,
  898. ) async {
  899. final db = await instance.database;
  900. return db.delete(
  901. table,
  902. where:
  903. '$columnCollectionID = ? AND $columnUploadedFileID IN (${uploadedFileIDs.join(', ')})',
  904. whereArgs: [collectionID],
  905. );
  906. }
  907. Future<int> collectionFileCount(int collectionID) async {
  908. final db = await instance.database;
  909. final count = Sqflite.firstIntValue(
  910. await db.rawQuery(
  911. 'SELECT COUNT(*) FROM $table where $columnCollectionID = $collectionID',
  912. ),
  913. );
  914. return count;
  915. }
  916. Future<int> fileCountWithVisibility(int visibility, int ownerID) async {
  917. final db = await instance.database;
  918. final count = Sqflite.firstIntValue(
  919. await db.rawQuery(
  920. 'SELECT COUNT(*) FROM $table where $columnMMdVisibility = $visibility AND $columnOwnerID = $ownerID',
  921. ),
  922. );
  923. return count;
  924. }
  925. Future<int> deleteCollection(int collectionID) async {
  926. final db = await instance.database;
  927. return db.delete(
  928. table,
  929. where: '$columnCollectionID = ?',
  930. whereArgs: [collectionID],
  931. );
  932. }
  933. Future<int> removeFromCollection(int collectionID, List<int> fileIDs) async {
  934. final db = await instance.database;
  935. return db.delete(
  936. table,
  937. where:
  938. '$columnCollectionID =? AND $columnUploadedFileID IN (${fileIDs.join(', ')})',
  939. whereArgs: [collectionID],
  940. );
  941. }
  942. Future<List<File>> getLatestLocalFiles() async {
  943. final db = await instance.database;
  944. final rows = await db.rawQuery(
  945. '''
  946. SELECT $table.*
  947. FROM $table
  948. INNER JOIN
  949. (
  950. SELECT $columnDeviceFolder, MAX($columnCreationTime) AS max_creation_time
  951. FROM $table
  952. WHERE $table.$columnLocalID IS NOT NULL
  953. GROUP BY $columnDeviceFolder
  954. ) latest_files
  955. ON $table.$columnDeviceFolder = latest_files.$columnDeviceFolder
  956. AND $table.$columnCreationTime = latest_files.max_creation_time;
  957. ''',
  958. );
  959. final files = _convertToFiles(rows);
  960. // TODO: Do this de-duplication within the SQL Query
  961. final folderMap = <String, File>{};
  962. for (final file in files) {
  963. if (folderMap.containsKey(file.deviceFolder)) {
  964. if (folderMap[file.deviceFolder].updationTime < file.updationTime) {
  965. continue;
  966. }
  967. }
  968. folderMap[file.deviceFolder] = file;
  969. }
  970. return folderMap.values.toList();
  971. }
  972. Future<List<File>> getLatestCollectionFiles() async {
  973. final db = await instance.database;
  974. final rows = await db.rawQuery(
  975. '''
  976. SELECT $table.*
  977. FROM $table
  978. INNER JOIN
  979. (
  980. SELECT $columnCollectionID, MAX($columnCreationTime) AS max_creation_time
  981. FROM $table
  982. WHERE ($columnCollectionID IS NOT NULL AND $columnCollectionID IS NOT -1)
  983. GROUP BY $columnCollectionID
  984. ) latest_files
  985. ON $table.$columnCollectionID = latest_files.$columnCollectionID
  986. AND $table.$columnCreationTime = latest_files.max_creation_time;
  987. ''',
  988. );
  989. final files = _convertToFiles(rows);
  990. // TODO: Do this de-duplication within the SQL Query
  991. final collectionMap = <int, File>{};
  992. for (final file in files) {
  993. if (collectionMap.containsKey(file.collectionID)) {
  994. if (collectionMap[file.collectionID].updationTime < file.updationTime) {
  995. continue;
  996. }
  997. }
  998. collectionMap[file.collectionID] = file;
  999. }
  1000. return collectionMap.values.toList();
  1001. }
  1002. Future<Map<String, int>> getFileCountInDeviceFolders() async {
  1003. final db = await instance.database;
  1004. final rows = await db.rawQuery(
  1005. '''
  1006. SELECT COUNT($columnGeneratedID) as count, $columnDeviceFolder
  1007. FROM $table
  1008. WHERE $columnLocalID IS NOT NULL
  1009. GROUP BY $columnDeviceFolder
  1010. ''',
  1011. );
  1012. final result = <String, int>{};
  1013. for (final row in rows) {
  1014. result[row[columnDeviceFolder]] = row["count"];
  1015. }
  1016. return result;
  1017. }
  1018. Future<List<String>> getLocalFilesBackedUpWithoutLocation() async {
  1019. final db = await instance.database;
  1020. final rows = await db.query(
  1021. table,
  1022. columns: [columnLocalID],
  1023. distinct: true,
  1024. where:
  1025. '$columnLocalID IS NOT NULL AND ($columnUploadedFileID IS NOT NULL AND $columnUploadedFileID IS NOT -1) '
  1026. 'AND ($columnLatitude IS NULL OR $columnLongitude IS NULL OR $columnLongitude = 0.0 or $columnLongitude = 0.0)',
  1027. );
  1028. final result = <String>[];
  1029. for (final row in rows) {
  1030. result.add(row[columnLocalID]);
  1031. }
  1032. return result;
  1033. }
  1034. Future<void> markForReUploadIfLocationMissing(List<String> localIDs) async {
  1035. if (localIDs.isEmpty) {
  1036. return;
  1037. }
  1038. String inParam = "";
  1039. for (final localID in localIDs) {
  1040. inParam += "'" + localID + "',";
  1041. }
  1042. inParam = inParam.substring(0, inParam.length - 1);
  1043. final db = await instance.database;
  1044. await db.rawUpdate(
  1045. '''
  1046. UPDATE $table
  1047. SET $columnUpdationTime = NULL
  1048. WHERE $columnLocalID IN ($inParam)
  1049. AND ($columnLatitude IS NULL OR $columnLongitude IS NULL OR $columnLongitude = 0.0 or $columnLongitude = 0.0);
  1050. ''',
  1051. );
  1052. }
  1053. Future<bool> doesFileExistInCollection(
  1054. int uploadedFileID,
  1055. int collectionID,
  1056. ) async {
  1057. final db = await instance.database;
  1058. final rows = await db.query(
  1059. table,
  1060. where: '$columnUploadedFileID = ? AND $columnCollectionID = ?',
  1061. whereArgs: [uploadedFileID, collectionID],
  1062. limit: 1,
  1063. );
  1064. return rows.isNotEmpty;
  1065. }
  1066. Future<Map<int, File>> getFilesFromIDs(List<int> ids) async {
  1067. final result = <int, File>{};
  1068. if (ids.isEmpty) {
  1069. return result;
  1070. }
  1071. String inParam = "";
  1072. for (final id in ids) {
  1073. inParam += "'" + id.toString() + "',";
  1074. }
  1075. inParam = inParam.substring(0, inParam.length - 1);
  1076. final db = await instance.database;
  1077. final results = await db.query(
  1078. table,
  1079. where: '$columnUploadedFileID IN ($inParam)',
  1080. );
  1081. final files = _convertToFiles(results);
  1082. for (final file in files) {
  1083. result[file.uploadedFileID] = file;
  1084. }
  1085. return result;
  1086. }
  1087. List<File> _convertToFiles(List<Map<String, dynamic>> results) {
  1088. final List<File> files = [];
  1089. for (final result in results) {
  1090. files.add(_getFileFromRow(result));
  1091. }
  1092. return files;
  1093. }
  1094. Future<List<File>> getAllFilesFromDB() async {
  1095. final db = await instance.database;
  1096. final List<Map<String, dynamic>> result = await db.query(table);
  1097. final List<File> files = _convertToFiles(result);
  1098. final List<File> deduplicatedFiles =
  1099. _deduplicatedAndFilterIgnoredFiles(files, null);
  1100. return deduplicatedFiles;
  1101. }
  1102. Map<String, dynamic> _getRowForFile(File file) {
  1103. final row = <String, dynamic>{};
  1104. if (file.generatedID != null) {
  1105. row[columnGeneratedID] = file.generatedID;
  1106. }
  1107. row[columnLocalID] = file.localID;
  1108. row[columnUploadedFileID] = file.uploadedFileID ?? -1;
  1109. row[columnOwnerID] = file.ownerID;
  1110. row[columnCollectionID] = file.collectionID ?? -1;
  1111. row[columnTitle] = file.title;
  1112. row[columnDeviceFolder] = file.deviceFolder;
  1113. if (file.location != null) {
  1114. row[columnLatitude] = file.location.latitude;
  1115. row[columnLongitude] = file.location.longitude;
  1116. }
  1117. row[columnFileType] = getInt(file.fileType);
  1118. row[columnCreationTime] = file.creationTime;
  1119. row[columnModificationTime] = file.modificationTime;
  1120. row[columnUpdationTime] = file.updationTime;
  1121. row[columnEncryptedKey] = file.encryptedKey;
  1122. row[columnKeyDecryptionNonce] = file.keyDecryptionNonce;
  1123. row[columnFileDecryptionHeader] = file.fileDecryptionHeader;
  1124. row[columnThumbnailDecryptionHeader] = file.thumbnailDecryptionHeader;
  1125. row[columnMetadataDecryptionHeader] = file.metadataDecryptionHeader;
  1126. row[columnFileSubType] = file.fileSubType ?? -1;
  1127. row[columnDuration] = file.duration ?? 0;
  1128. row[columnExif] = file.exif;
  1129. row[columnHash] = file.hash;
  1130. row[columnMetadataVersion] = file.metadataVersion;
  1131. row[columnMMdVersion] = file.mMdVersion ?? 0;
  1132. row[columnMMdEncodedJson] = file.mMdEncodedJson ?? '{}';
  1133. row[columnMMdVisibility] =
  1134. file.magicMetadata?.visibility ?? kVisibilityVisible;
  1135. row[columnPubMMdVersion] = file.pubMmdVersion ?? 0;
  1136. row[columnPubMMdEncodedJson] = file.pubMmdEncodedJson ?? '{}';
  1137. if (file.pubMagicMetadata != null &&
  1138. file.pubMagicMetadata.editedTime != null) {
  1139. // override existing creationTime to avoid re-writing all queries related
  1140. // to loading the gallery
  1141. row[columnCreationTime] = file.pubMagicMetadata.editedTime;
  1142. }
  1143. return row;
  1144. }
  1145. Map<String, dynamic> _getRowForFileWithoutCollection(File file) {
  1146. final row = <String, dynamic>{};
  1147. row[columnLocalID] = file.localID;
  1148. row[columnUploadedFileID] = file.uploadedFileID ?? -1;
  1149. row[columnOwnerID] = file.ownerID;
  1150. row[columnTitle] = file.title;
  1151. row[columnDeviceFolder] = file.deviceFolder;
  1152. if (file.location != null) {
  1153. row[columnLatitude] = file.location.latitude;
  1154. row[columnLongitude] = file.location.longitude;
  1155. }
  1156. row[columnFileType] = getInt(file.fileType);
  1157. row[columnCreationTime] = file.creationTime;
  1158. row[columnModificationTime] = file.modificationTime;
  1159. row[columnUpdationTime] = file.updationTime;
  1160. row[columnFileDecryptionHeader] = file.fileDecryptionHeader;
  1161. row[columnThumbnailDecryptionHeader] = file.thumbnailDecryptionHeader;
  1162. row[columnMetadataDecryptionHeader] = file.metadataDecryptionHeader;
  1163. row[columnFileSubType] = file.fileSubType ?? -1;
  1164. row[columnDuration] = file.duration ?? 0;
  1165. row[columnExif] = file.exif;
  1166. row[columnHash] = file.hash;
  1167. row[columnMetadataVersion] = file.metadataVersion;
  1168. row[columnMMdVersion] = file.mMdVersion ?? 0;
  1169. row[columnMMdEncodedJson] = file.mMdEncodedJson ?? '{}';
  1170. row[columnMMdVisibility] =
  1171. file.magicMetadata?.visibility ?? kVisibilityVisible;
  1172. row[columnPubMMdVersion] = file.pubMmdVersion ?? 0;
  1173. row[columnPubMMdEncodedJson] = file.pubMmdEncodedJson ?? '{}';
  1174. if (file.pubMagicMetadata != null &&
  1175. file.pubMagicMetadata.editedTime != null) {
  1176. // override existing creationTime to avoid re-writing all queries related
  1177. // to loading the gallery
  1178. row[columnCreationTime] = file.pubMagicMetadata.editedTime;
  1179. }
  1180. return row;
  1181. }
  1182. File _getFileFromRow(Map<String, dynamic> row) {
  1183. final file = File();
  1184. file.generatedID = row[columnGeneratedID];
  1185. file.localID = row[columnLocalID];
  1186. file.uploadedFileID =
  1187. row[columnUploadedFileID] == -1 ? null : row[columnUploadedFileID];
  1188. file.ownerID = row[columnOwnerID];
  1189. file.collectionID =
  1190. row[columnCollectionID] == -1 ? null : row[columnCollectionID];
  1191. file.title = row[columnTitle];
  1192. file.deviceFolder = row[columnDeviceFolder];
  1193. if (row[columnLatitude] != null && row[columnLongitude] != null) {
  1194. file.location = Location(row[columnLatitude], row[columnLongitude]);
  1195. }
  1196. file.fileType = getFileType(row[columnFileType]);
  1197. file.creationTime = row[columnCreationTime];
  1198. file.modificationTime = row[columnModificationTime];
  1199. file.updationTime = row[columnUpdationTime] ?? -1;
  1200. file.encryptedKey = row[columnEncryptedKey];
  1201. file.keyDecryptionNonce = row[columnKeyDecryptionNonce];
  1202. file.fileDecryptionHeader = row[columnFileDecryptionHeader];
  1203. file.thumbnailDecryptionHeader = row[columnThumbnailDecryptionHeader];
  1204. file.metadataDecryptionHeader = row[columnMetadataDecryptionHeader];
  1205. file.fileSubType = row[columnFileSubType] ?? -1;
  1206. file.duration = row[columnDuration] ?? 0;
  1207. file.exif = row[columnExif];
  1208. file.hash = row[columnHash];
  1209. file.metadataVersion = row[columnMetadataVersion] ?? 0;
  1210. file.mMdVersion = row[columnMMdVersion] ?? 0;
  1211. file.mMdEncodedJson = row[columnMMdEncodedJson] ?? '{}';
  1212. file.pubMmdVersion = row[columnPubMMdVersion] ?? 0;
  1213. file.pubMmdEncodedJson = row[columnPubMMdEncodedJson] ?? '{}';
  1214. return file;
  1215. }
  1216. }