import 'dart:convert'; import 'package:drift/drift.dart'; import 'package:nc_photos/ci_string.dart'; import 'package:nc_photos/entity/album.dart'; import 'package:nc_photos/entity/album/cover_provider.dart'; import 'package:nc_photos/entity/album/provider.dart'; import 'package:nc_photos/entity/album/sort_provider.dart'; import 'package:nc_photos/entity/exif.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/entity/file_descriptor.dart'; import 'package:nc_photos/entity/person.dart'; import 'package:nc_photos/entity/sqlite_table.dart' as sql; import 'package:nc_photos/entity/sqlite_table_extension.dart' as sql; import 'package:nc_photos/entity/tag.dart'; import 'package:nc_photos/iterable_extension.dart'; import 'package:nc_photos/object_extension.dart'; import 'package:nc_photos/or_null.dart'; extension SqlTagListExtension on List { Future> convertToAppTag() { return computeAll(SqliteTagConverter.fromSql); } } extension AppTagListExtension on List { Future> convertToTagCompanion( sql.Account? dbAccount) { return map((t) => { "account": dbAccount, "tag": t, }).computeAll(_convertAppTag); } } extension SqlPersonListExtension on List { Future> convertToAppPerson() { return computeAll(SqlitePersonConverter.fromSql); } } extension AppPersonListExtension on List { Future> convertToPersonCompanion( sql.Account? dbAccount) { return map((p) => { "account": dbAccount, "person": p, }).computeAll(_convertAppPerson); } } class SqliteAlbumConverter { static Album fromSql( sql.Album album, File albumFile, List shares) { return Album( lastUpdated: album.lastUpdated, name: album.name, provider: AlbumProvider.fromJson({ "type": album.providerType, "content": jsonDecode(album.providerContent), }), coverProvider: AlbumCoverProvider.fromJson({ "type": album.coverProviderType, "content": jsonDecode(album.coverProviderContent), }), sortProvider: AlbumSortProvider.fromJson({ "type": album.sortProviderType, "content": jsonDecode(album.sortProviderContent), }), shares: shares.isEmpty ? null : shares .map((e) => AlbumShare( userId: e.userId.toCi(), displayName: e.displayName, sharedAt: e.sharedAt.toUtc(), )) .toList(), // replace with the original etag when this album was cached albumFile: albumFile.copyWith(etag: OrNull(album.fileEtag)), savedVersion: album.version, ); } static sql.CompleteAlbumCompanion toSql( Album album, int albumFileRowId, String albumFileEtag) { final providerJson = album.provider.toJson(); final coverProviderJson = album.coverProvider.toJson(); final sortProviderJson = album.sortProvider.toJson(); final dbAlbum = sql.AlbumsCompanion.insert( file: albumFileRowId, fileEtag: Value(albumFileEtag), version: Album.version, lastUpdated: album.lastUpdated, name: album.name, providerType: providerJson["type"], providerContent: jsonEncode(providerJson["content"]), coverProviderType: coverProviderJson["type"], coverProviderContent: jsonEncode(coverProviderJson["content"]), sortProviderType: sortProviderJson["type"], sortProviderContent: jsonEncode(sortProviderJson["content"]), ); final dbAlbumShares = album.shares ?.map((s) => sql.AlbumSharesCompanion( userId: Value(s.userId.toCaseInsensitiveString()), displayName: Value(s.displayName), sharedAt: Value(s.sharedAt), )) .toList(); return sql.CompleteAlbumCompanion(dbAlbum, dbAlbumShares ?? []); } } class SqliteFileConverter { static File fromSql(String userId, sql.CompleteFile f) { final metadata = f.image?.run((obj) => Metadata( lastUpdated: obj.lastUpdated, fileEtag: obj.fileEtag, imageWidth: obj.width, imageHeight: obj.height, exif: obj.exifRaw?.run((e) => Exif.fromJson(jsonDecode(e))), )); final location = f.imageLocation?.run((obj) => ImageLocation( version: obj.version, name: obj.name, latitude: obj.latitude, longitude: obj.longitude, countryCode: obj.countryCode, admin1: obj.admin1, admin2: obj.admin2, )); return File( path: "remote.php/dav/files/$userId/${f.accountFile.relativePath}", contentLength: f.file.contentLength, contentType: f.file.contentType, etag: f.file.etag, lastModified: f.file.lastModified, isCollection: f.file.isCollection, usedBytes: f.file.usedBytes, hasPreview: f.file.hasPreview, fileId: f.file.fileId, isFavorite: f.accountFile.isFavorite, ownerId: f.file.ownerId?.toCi(), ownerDisplayName: f.file.ownerDisplayName, trashbinFilename: f.trash?.filename, trashbinOriginalLocation: f.trash?.originalLocation, trashbinDeletionTime: f.trash?.deletionTime, metadata: metadata, isArchived: f.accountFile.isArchived, overrideDateTime: f.accountFile.overrideDateTime, location: location, ); } static sql.CompleteFileCompanion toSql(sql.Account? account, File file) { final dbFile = sql.FilesCompanion( server: account == null ? const Value.absent() : Value(account.server), fileId: Value(file.fileId!), contentLength: Value(file.contentLength), contentType: Value(file.contentType), etag: Value(file.etag), lastModified: Value(file.lastModified), isCollection: Value(file.isCollection), usedBytes: Value(file.usedBytes), hasPreview: Value(file.hasPreview), ownerId: Value(file.ownerId!.toCaseInsensitiveString()), ownerDisplayName: Value(file.ownerDisplayName), ); final dbAccountFile = sql.AccountFilesCompanion( account: account == null ? const Value.absent() : Value(account.rowId), relativePath: Value(file.strippedPathWithEmpty), isFavorite: Value(file.isFavorite), isArchived: Value(file.isArchived), overrideDateTime: Value(file.overrideDateTime), bestDateTime: Value(file.bestDateTime), ); final dbImage = file.metadata?.run((m) => sql.ImagesCompanion.insert( lastUpdated: m.lastUpdated, fileEtag: Value(m.fileEtag), width: Value(m.imageWidth), height: Value(m.imageHeight), exifRaw: Value(m.exif?.toJson().run((j) => jsonEncode(j))), dateTimeOriginal: Value(m.exif?.dateTimeOriginal), )); final dbImageLocation = file.location?.run((l) => sql.ImageLocationsCompanion.insert( version: l.version, name: Value(l.name), latitude: Value(l.latitude), longitude: Value(l.longitude), countryCode: Value(l.countryCode), admin1: Value(l.admin1), admin2: Value(l.admin2), )); final dbTrash = file.trashbinDeletionTime == null ? null : sql.TrashesCompanion.insert( filename: file.trashbinFilename!, originalLocation: file.trashbinOriginalLocation!, deletionTime: file.trashbinDeletionTime!, ); return sql.CompleteFileCompanion( dbFile, dbAccountFile, dbImage, dbImageLocation, dbTrash); } } class SqliteTagConverter { static Tag fromSql(sql.Tag tag) => Tag( id: tag.tagId, displayName: tag.displayName, userVisible: tag.userVisible, userAssignable: tag.userAssignable, ); static sql.TagsCompanion toSql(sql.Account? dbAccount, Tag tag) => sql.TagsCompanion( server: dbAccount == null ? const Value.absent() : Value(dbAccount.server), tagId: Value(tag.id), displayName: Value(tag.displayName), userVisible: Value(tag.userVisible), userAssignable: Value(tag.userAssignable), ); } class SqlitePersonConverter { static Person fromSql(sql.Person person) => Person( name: person.name, thumbFaceId: person.thumbFaceId, count: person.count, ); static sql.PersonsCompanion toSql(sql.Account? dbAccount, Person person) => sql.PersonsCompanion( account: dbAccount == null ? const Value.absent() : Value(dbAccount.rowId), name: Value(person.name), thumbFaceId: Value(person.thumbFaceId), count: Value(person.count), ); } sql.TagsCompanion _convertAppTag(Map map) { final account = map["account"] as sql.Account?; final tag = map["tag"] as Tag; return SqliteTagConverter.toSql(account, tag); } sql.PersonsCompanion _convertAppPerson(Map map) { final account = map["account"] as sql.Account?; final person = map["person"] as Person; return SqlitePersonConverter.toSql(account, person); }