2021-04-10 06:28:12 +02:00
|
|
|
import 'dart:typed_data';
|
|
|
|
|
2021-04-15 20:44:25 +02:00
|
|
|
import 'package:equatable/equatable.dart';
|
2021-04-10 06:28:12 +02:00
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
|
|
|
import 'package:nc_photos/entity/exif.dart';
|
2021-04-28 21:22:33 +02:00
|
|
|
import 'package:nc_photos/or_null.dart';
|
2021-04-10 06:28:12 +02:00
|
|
|
import 'package:nc_photos/string_extension.dart';
|
|
|
|
|
|
|
|
int compareFileDateTimeDescending(File x, File y) {
|
|
|
|
final xDate = x.metadata?.exif?.dateTimeOriginal ?? x.lastModified;
|
|
|
|
final yDate = y.metadata?.exif?.dateTimeOriginal ?? y.lastModified;
|
|
|
|
final tmp = yDate.compareTo(xDate);
|
|
|
|
if (tmp != 0) {
|
|
|
|
return tmp;
|
|
|
|
} else {
|
|
|
|
// compare file name if files are modified at the same time
|
|
|
|
return x.path.compareTo(y.path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Immutable object that hold metadata of a [File]
|
2021-04-15 20:44:25 +02:00
|
|
|
class Metadata with EquatableMixin {
|
2021-04-10 06:28:12 +02:00
|
|
|
Metadata({
|
|
|
|
DateTime lastUpdated,
|
|
|
|
this.fileEtag,
|
|
|
|
this.imageWidth,
|
|
|
|
this.imageHeight,
|
|
|
|
this.exif,
|
|
|
|
}) : this.lastUpdated = (lastUpdated ?? DateTime.now()).toUtc();
|
|
|
|
|
|
|
|
/// Parse Metadata from [json]
|
|
|
|
///
|
|
|
|
/// If the version saved in json does not match the active one, the
|
|
|
|
/// corresponding upgrader will be called one by one to upgrade the json,
|
|
|
|
/// version by version until it reached the active version. If any upgrader
|
|
|
|
/// in the chain is null, the upgrade process will fail
|
|
|
|
factory Metadata.fromJson(
|
|
|
|
Map<String, dynamic> json, {
|
|
|
|
MetadataUpgraderV1 upgraderV1,
|
2021-04-15 00:57:23 +02:00
|
|
|
MetadataUpgraderV2 upgraderV2,
|
2021-04-10 06:28:12 +02:00
|
|
|
}) {
|
|
|
|
final jsonVersion = json["version"];
|
|
|
|
if (jsonVersion < 2) {
|
|
|
|
json = upgraderV1?.call(json);
|
|
|
|
if (json == null) {
|
|
|
|
_log.info("[fromJson] Version $jsonVersion not compatible");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2021-04-15 00:57:23 +02:00
|
|
|
if (jsonVersion < 3) {
|
|
|
|
json = upgraderV2?.call(json);
|
|
|
|
if (json == null) {
|
|
|
|
_log.info("[fromJson] Version $jsonVersion not compatible");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
return Metadata(
|
|
|
|
lastUpdated: json["lastUpdated"] == null
|
|
|
|
? null
|
|
|
|
: DateTime.parse(json["lastUpdated"]),
|
|
|
|
fileEtag: json["fileEtag"],
|
|
|
|
imageWidth: json["imageWidth"],
|
|
|
|
imageHeight: json["imageHeight"],
|
|
|
|
exif: json["exif"] == null
|
|
|
|
? null
|
|
|
|
: Exif.fromJson(json["exif"].cast<String, dynamic>()),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Map<String, dynamic> toJson() {
|
|
|
|
return {
|
|
|
|
"version": version,
|
|
|
|
"lastUpdated": lastUpdated.toIso8601String(),
|
|
|
|
if (fileEtag != null) "fileEtag": fileEtag,
|
|
|
|
if (imageWidth != null) "imageWidth": imageWidth,
|
|
|
|
if (imageHeight != null) "imageHeight": imageHeight,
|
|
|
|
if (exif != null) "exif": exif.toJson(),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
toString() {
|
|
|
|
var product = "$runtimeType {"
|
|
|
|
"lastUpdated: $lastUpdated, ";
|
|
|
|
if (fileEtag != null) {
|
|
|
|
product += "fileEtag: $fileEtag, ";
|
|
|
|
}
|
|
|
|
if (imageWidth != null) {
|
|
|
|
product += "imageWidth: $imageWidth, ";
|
|
|
|
}
|
|
|
|
if (imageHeight != null) {
|
|
|
|
product += "imageHeight: $imageHeight, ";
|
|
|
|
}
|
|
|
|
if (exif != null) {
|
|
|
|
product += "exif: $exif, ";
|
|
|
|
}
|
|
|
|
return product + "}";
|
|
|
|
}
|
|
|
|
|
2021-04-15 20:44:25 +02:00
|
|
|
@override
|
|
|
|
get props => [
|
|
|
|
lastUpdated,
|
|
|
|
fileEtag,
|
|
|
|
imageWidth,
|
|
|
|
imageHeight,
|
|
|
|
exif,
|
|
|
|
];
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
final DateTime lastUpdated;
|
|
|
|
|
|
|
|
/// Etag of the parent file when the metadata is saved
|
|
|
|
final String fileEtag;
|
|
|
|
final int imageWidth;
|
|
|
|
final int imageHeight;
|
|
|
|
final Exif exif;
|
|
|
|
|
|
|
|
/// versioning of this class, use to upgrade old persisted metadata
|
2021-04-15 00:57:23 +02:00
|
|
|
static const version = 3;
|
2021-04-10 06:28:12 +02:00
|
|
|
|
|
|
|
static final _log = Logger("entity.file.Metadata");
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract class MetadataUpgrader {
|
|
|
|
Map<String, dynamic> call(Map<String, dynamic> json);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Upgrade v1 Metadata to v2
|
|
|
|
class MetadataUpgraderV1 implements MetadataUpgrader {
|
|
|
|
MetadataUpgraderV1({
|
|
|
|
@required this.fileContentType,
|
2021-04-15 00:57:23 +02:00
|
|
|
this.logFilePath,
|
2021-04-10 06:28:12 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
Map<String, dynamic> call(Map<String, dynamic> json) {
|
|
|
|
if (fileContentType == "image/webp") {
|
|
|
|
// Version 1 metadata for webp is bugged, drop it
|
2021-04-15 00:57:23 +02:00
|
|
|
_log.fine("[call] Upgrade v1 metadata for file: $logFilePath");
|
2021-04-10 06:28:12 +02:00
|
|
|
return null;
|
|
|
|
} else {
|
|
|
|
return json;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final String fileContentType;
|
2021-04-15 00:57:23 +02:00
|
|
|
|
|
|
|
/// File path for logging only
|
|
|
|
final String logFilePath;
|
|
|
|
|
|
|
|
static final _log = Logger("entity.file.MetadataUpgraderV1");
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Upgrade v2 Metadata to v3
|
|
|
|
class MetadataUpgraderV2 implements MetadataUpgrader {
|
|
|
|
MetadataUpgraderV2({
|
|
|
|
@required this.fileContentType,
|
|
|
|
this.logFilePath,
|
|
|
|
});
|
|
|
|
|
|
|
|
Map<String, dynamic> call(Map<String, dynamic> json) {
|
|
|
|
if (fileContentType == "image/jpeg") {
|
|
|
|
// Version 2 metadata for jpeg doesn't consider orientation
|
|
|
|
if (json["exif"] != null && json["exif"].containsKey("Orientation")) {
|
|
|
|
// Check orientation
|
|
|
|
final orientation = json["exif"]["Orientation"];
|
|
|
|
if (orientation >= 5 && orientation <= 8) {
|
|
|
|
_log.fine("[call] Upgrade v2 metadata for file: $logFilePath");
|
|
|
|
final temp = json["imageWidth"];
|
|
|
|
json["imageWidth"] = json["imageHeight"];
|
|
|
|
json["imageHeight"] = temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return json;
|
|
|
|
}
|
|
|
|
|
|
|
|
final String fileContentType;
|
|
|
|
|
|
|
|
/// File path for logging only
|
|
|
|
final String logFilePath;
|
|
|
|
|
|
|
|
static final _log = Logger("entity.file.MetadataUpgraderV2");
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
|
2021-04-15 20:44:25 +02:00
|
|
|
class File with EquatableMixin {
|
2021-04-10 06:28:12 +02:00
|
|
|
File({
|
|
|
|
@required String path,
|
|
|
|
this.contentLength,
|
|
|
|
this.contentType,
|
|
|
|
this.etag,
|
|
|
|
this.lastModified,
|
|
|
|
this.isCollection,
|
|
|
|
this.usedBytes,
|
|
|
|
this.hasPreview,
|
2021-05-10 07:53:05 +02:00
|
|
|
this.fileId,
|
2021-04-10 06:28:12 +02:00
|
|
|
this.metadata,
|
2021-05-25 22:10:43 +02:00
|
|
|
}) : this.path = path.trimAny("/");
|
2021-04-10 06:28:12 +02:00
|
|
|
|
|
|
|
factory File.fromJson(Map<String, dynamic> json) {
|
|
|
|
return File(
|
|
|
|
path: json["path"],
|
|
|
|
contentLength: json["contentLength"],
|
|
|
|
contentType: json["contentType"],
|
|
|
|
etag: json["etag"],
|
|
|
|
lastModified: json["lastModified"] == null
|
|
|
|
? null
|
|
|
|
: DateTime.parse(json["lastModified"]),
|
|
|
|
isCollection: json["isCollection"],
|
|
|
|
usedBytes: json["usedBytes"],
|
|
|
|
hasPreview: json["hasPreview"],
|
2021-05-10 07:53:05 +02:00
|
|
|
fileId: json["fileId"],
|
2021-04-10 06:28:12 +02:00
|
|
|
metadata: json["metadata"] == null
|
|
|
|
? null
|
|
|
|
: Metadata.fromJson(
|
|
|
|
json["metadata"].cast<String, dynamic>(),
|
|
|
|
upgraderV1: MetadataUpgraderV1(
|
|
|
|
fileContentType: json["contentType"],
|
2021-04-15 00:57:23 +02:00
|
|
|
logFilePath: json["path"],
|
|
|
|
),
|
|
|
|
upgraderV2: MetadataUpgraderV2(
|
|
|
|
fileContentType: json["contentType"],
|
|
|
|
logFilePath: json["path"],
|
2021-04-10 06:28:12 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
toString() {
|
|
|
|
var product = "$runtimeType {"
|
|
|
|
"path: '$path', ";
|
|
|
|
if (contentLength != null) {
|
|
|
|
product += "contentLength: $contentLength, ";
|
|
|
|
}
|
|
|
|
if (contentType != null) {
|
|
|
|
product += "contentType: '$contentType', ";
|
|
|
|
}
|
|
|
|
if (etag != null) {
|
|
|
|
product += "etag: '$etag', ";
|
|
|
|
}
|
|
|
|
if (lastModified != null) {
|
|
|
|
product += "lastModified: $lastModified, ";
|
|
|
|
}
|
|
|
|
if (isCollection != null) {
|
|
|
|
product += "isCollection: $isCollection, ";
|
|
|
|
}
|
|
|
|
if (usedBytes != null) {
|
|
|
|
product += "usedBytes: $usedBytes, ";
|
|
|
|
}
|
|
|
|
if (hasPreview != null) {
|
|
|
|
product += "hasPreview: $hasPreview, ";
|
|
|
|
}
|
2021-05-10 07:53:05 +02:00
|
|
|
if (fileId != null) {
|
|
|
|
product += "fileId: '$fileId', ";
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
if (metadata != null) {
|
|
|
|
product += "metadata: $metadata, ";
|
|
|
|
}
|
|
|
|
return product + "}";
|
|
|
|
}
|
|
|
|
|
|
|
|
Map<String, dynamic> toJson() {
|
|
|
|
return {
|
|
|
|
"path": path,
|
|
|
|
if (contentLength != null) "contentLength": contentLength,
|
|
|
|
if (contentType != null) "contentType": contentType,
|
|
|
|
if (etag != null) "etag": etag,
|
|
|
|
if (lastModified != null) "lastModified": lastModified.toIso8601String(),
|
|
|
|
if (isCollection != null) "isCollection": isCollection,
|
|
|
|
if (usedBytes != null) "usedBytes": usedBytes,
|
|
|
|
if (hasPreview != null) "hasPreview": hasPreview,
|
2021-05-10 07:53:05 +02:00
|
|
|
if (fileId != null) "fileId": fileId,
|
2021-04-10 06:28:12 +02:00
|
|
|
if (metadata != null) "metadata": metadata.toJson(),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
File copyWith({
|
|
|
|
String path,
|
|
|
|
int contentLength,
|
|
|
|
String contentType,
|
|
|
|
String etag,
|
|
|
|
DateTime lastModified,
|
|
|
|
bool isCollection,
|
|
|
|
int usedBytes,
|
|
|
|
bool hasPreview,
|
2021-05-10 07:53:05 +02:00
|
|
|
int fileId,
|
2021-04-28 21:22:33 +02:00
|
|
|
OrNull<Metadata> metadata,
|
2021-04-10 06:28:12 +02:00
|
|
|
}) {
|
|
|
|
return File(
|
|
|
|
path: path ?? this.path,
|
|
|
|
contentLength: contentLength ?? this.contentLength,
|
|
|
|
contentType: contentType ?? this.contentType,
|
|
|
|
etag: etag ?? this.etag,
|
|
|
|
lastModified: lastModified ?? this.lastModified,
|
|
|
|
isCollection: isCollection ?? this.isCollection,
|
|
|
|
usedBytes: usedBytes ?? this.usedBytes,
|
|
|
|
hasPreview: hasPreview ?? this.hasPreview,
|
2021-05-10 07:53:05 +02:00
|
|
|
fileId: fileId ?? this.fileId,
|
2021-04-28 21:22:33 +02:00
|
|
|
metadata: metadata == null ? this.metadata : metadata.obj,
|
2021-04-10 06:28:12 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the path of this file with the DAV part stripped
|
2021-05-25 08:58:18 +02:00
|
|
|
///
|
|
|
|
/// WebDAV file path: remote.php/dav/files/{username}/{strippedPath}
|
2021-04-10 06:28:12 +02:00
|
|
|
String get strippedPath {
|
|
|
|
if (path.contains("remote.php/dav/files")) {
|
2021-05-25 08:58:18 +02:00
|
|
|
final position = path.indexOf("/", "remote.php/dav/files/".length) + 1;
|
|
|
|
if (position == 0) {
|
|
|
|
// root dir path
|
|
|
|
return ".";
|
|
|
|
} else {
|
|
|
|
return path.substring(position);
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
} else {
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-15 20:44:25 +02:00
|
|
|
@override
|
|
|
|
get props => [
|
|
|
|
path,
|
|
|
|
contentLength,
|
|
|
|
contentType,
|
|
|
|
etag,
|
|
|
|
lastModified,
|
|
|
|
isCollection,
|
|
|
|
usedBytes,
|
|
|
|
hasPreview,
|
2021-05-10 07:53:05 +02:00
|
|
|
fileId,
|
2021-04-15 20:44:25 +02:00
|
|
|
metadata,
|
|
|
|
];
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
final String path;
|
|
|
|
final int contentLength;
|
|
|
|
final String contentType;
|
|
|
|
final String etag;
|
|
|
|
final DateTime lastModified;
|
|
|
|
final bool isCollection;
|
|
|
|
final int usedBytes;
|
|
|
|
final bool hasPreview;
|
2021-05-10 07:53:05 +02:00
|
|
|
// maybe null when loaded from old cache
|
|
|
|
final int fileId;
|
2021-04-10 06:28:12 +02:00
|
|
|
// metadata
|
|
|
|
final Metadata metadata;
|
|
|
|
}
|
|
|
|
|
|
|
|
class FileRepo {
|
|
|
|
FileRepo(this.dataSrc);
|
|
|
|
|
|
|
|
/// See [FileDataSource.list]
|
|
|
|
Future<List<File>> list(Account account, File root) =>
|
|
|
|
this.dataSrc.list(account, root);
|
|
|
|
|
|
|
|
/// See [FileDataSource.remove]
|
|
|
|
Future<void> remove(Account account, File file) =>
|
|
|
|
this.dataSrc.remove(account, file);
|
|
|
|
|
|
|
|
/// See [FileDataSource.getBinary]
|
|
|
|
Future<Uint8List> getBinary(Account account, File file) =>
|
|
|
|
this.dataSrc.getBinary(account, file);
|
|
|
|
|
|
|
|
/// See [FileDataSource.putBinary]
|
|
|
|
Future<void> putBinary(Account account, String path, Uint8List content) =>
|
|
|
|
this.dataSrc.putBinary(account, path, content);
|
|
|
|
|
|
|
|
/// See [FileDataSource.updateMetadata]
|
|
|
|
Future<void> updateMetadata(Account account, File file, Metadata metadata) =>
|
|
|
|
this.dataSrc.updateMetadata(account, file, metadata);
|
|
|
|
|
2021-05-20 17:43:53 +02:00
|
|
|
/// See [FileDataSource.copy]
|
|
|
|
Future<void> copy(
|
|
|
|
Account account,
|
|
|
|
File f,
|
|
|
|
String destination, {
|
|
|
|
bool shouldOverwrite,
|
|
|
|
}) =>
|
|
|
|
this.dataSrc.copy(
|
|
|
|
account,
|
|
|
|
f,
|
|
|
|
destination,
|
|
|
|
shouldOverwrite: shouldOverwrite,
|
|
|
|
);
|
|
|
|
|
2021-05-20 17:45:06 +02:00
|
|
|
/// See [FileDataSource.move]
|
|
|
|
Future<void> move(
|
|
|
|
Account account,
|
|
|
|
File f,
|
|
|
|
String destination, {
|
|
|
|
bool shouldOverwrite,
|
|
|
|
}) =>
|
|
|
|
this.dataSrc.move(
|
|
|
|
account,
|
|
|
|
f,
|
|
|
|
destination,
|
|
|
|
shouldOverwrite: shouldOverwrite,
|
|
|
|
);
|
|
|
|
|
2021-05-20 17:47:24 +02:00
|
|
|
/// See [FileDataSource.createDir]
|
|
|
|
Future<void> createDir(Account account, String path) =>
|
|
|
|
this.dataSrc.createDir(account, path);
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
final FileDataSource dataSrc;
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract class FileDataSource {
|
|
|
|
/// List all files under [f]
|
|
|
|
Future<List<File>> list(Account account, File f);
|
|
|
|
|
|
|
|
/// Remove file
|
|
|
|
Future<void> remove(Account account, File f);
|
|
|
|
|
|
|
|
/// Read file as binary array
|
|
|
|
Future<Uint8List> getBinary(Account account, File f);
|
|
|
|
|
|
|
|
/// Upload content to [path]
|
|
|
|
Future<void> putBinary(Account account, String path, Uint8List content);
|
|
|
|
|
|
|
|
/// Update metadata for a file
|
|
|
|
///
|
|
|
|
/// This will completely replace the metadata of the file [f]. Partial update
|
|
|
|
/// is not supported
|
|
|
|
Future<void> updateMetadata(Account account, File f, Metadata metadata);
|
2021-05-20 17:43:53 +02:00
|
|
|
|
|
|
|
/// Copy [f] to [destination]
|
|
|
|
///
|
|
|
|
/// [destination] should be a relative WebDAV path like
|
|
|
|
/// remote.php/dav/files/admin/new/location
|
|
|
|
Future<void> copy(
|
|
|
|
Account account,
|
|
|
|
File f,
|
|
|
|
String destination, {
|
|
|
|
bool shouldOverwrite,
|
|
|
|
});
|
2021-05-20 17:45:06 +02:00
|
|
|
|
|
|
|
/// Move [f] to [destination]
|
|
|
|
///
|
|
|
|
/// [destination] should be a relative WebDAV path like
|
|
|
|
/// remote.php/dav/files/admin/new/location
|
|
|
|
Future<void> move(
|
|
|
|
Account account,
|
|
|
|
File f,
|
|
|
|
String destination, {
|
|
|
|
bool shouldOverwrite,
|
|
|
|
});
|
2021-05-20 17:47:24 +02:00
|
|
|
|
|
|
|
/// Create a directory at [path]
|
|
|
|
///
|
|
|
|
/// [path] should be a relative WebDAV path like
|
|
|
|
/// remote.php/dav/files/admin/new/dir
|
|
|
|
Future<void> createDir(Account account, String path);
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|