2021-04-10 06:28:12 +02:00
|
|
|
/// Helper functions working with remote Nextcloud server
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
|
|
|
import 'package:nc_photos/api/api.dart';
|
|
|
|
import 'package:nc_photos/entity/file.dart';
|
2021-08-01 22:46:16 +02:00
|
|
|
import 'package:nc_photos/entity/file_util.dart' as file_util;
|
2021-05-06 06:57:20 +02:00
|
|
|
import 'package:nc_photos/exception.dart';
|
2021-04-10 06:28:12 +02:00
|
|
|
|
2021-10-08 20:39:53 +02:00
|
|
|
/// Characters that are not allowed in filename
|
|
|
|
const reservedFilenameChars = "<>:\"/\\|?*";
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
/// Return the preview image URL for [file]. See [getFilePreviewUrlRelative]
|
|
|
|
String getFilePreviewUrl(
|
|
|
|
Account account,
|
|
|
|
File file, {
|
2021-07-23 22:05:57 +02:00
|
|
|
required int width,
|
|
|
|
required int height,
|
|
|
|
String? mode,
|
|
|
|
bool? a,
|
2021-04-10 06:28:12 +02:00
|
|
|
}) {
|
|
|
|
return "${account.url}/"
|
2021-08-01 22:46:16 +02:00
|
|
|
"${getFilePreviewUrlRelative(account, file, width: width, height: height, mode: mode, a: a)}";
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the relative preview image URL for [file]. If [a] == true, the
|
|
|
|
/// preview will maintain the original aspect ratio, otherwise it will be
|
|
|
|
/// cropped
|
|
|
|
String getFilePreviewUrlRelative(
|
2021-08-01 22:46:16 +02:00
|
|
|
Account account,
|
2021-04-10 06:28:12 +02:00
|
|
|
File file, {
|
2021-07-23 22:05:57 +02:00
|
|
|
required int width,
|
|
|
|
required int height,
|
|
|
|
String? mode,
|
|
|
|
bool? a,
|
2021-04-10 06:28:12 +02:00
|
|
|
}) {
|
2021-06-14 12:50:54 +02:00
|
|
|
String url;
|
2021-08-01 22:46:16 +02:00
|
|
|
if (file_util.isTrash(account, file)) {
|
|
|
|
// trashbin does not support preview.png endpoint
|
|
|
|
url = "index.php/apps/files_trashbin/preview?fileId=${file.fileId}";
|
2021-06-14 12:50:54 +02:00
|
|
|
} else {
|
2021-08-01 22:46:16 +02:00
|
|
|
if (file.fileId != null) {
|
|
|
|
url = "index.php/core/preview?fileId=${file.fileId}";
|
|
|
|
} else {
|
|
|
|
final filePath = Uri.encodeQueryComponent(file.strippedPath);
|
|
|
|
url = "index.php/core/preview.png?file=$filePath";
|
|
|
|
}
|
2021-06-14 12:50:54 +02:00
|
|
|
}
|
2021-08-01 22:46:16 +02:00
|
|
|
|
2021-06-14 12:50:54 +02:00
|
|
|
url = "$url&x=$width&y=$height";
|
2021-04-10 06:28:12 +02:00
|
|
|
if (mode != null) {
|
|
|
|
url = "$url&mode=$mode";
|
|
|
|
}
|
|
|
|
if (a != null) {
|
|
|
|
url = "$url&a=${a ? 1 : 0}";
|
|
|
|
}
|
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
|
|
|
String getFileUrl(Account account, File file) {
|
|
|
|
return "${account.url}/${getFileUrlRelative(file)}";
|
|
|
|
}
|
|
|
|
|
|
|
|
String getFileUrlRelative(File file) {
|
|
|
|
return file.path;
|
|
|
|
}
|
|
|
|
|
|
|
|
String getWebdavRootUrlRelative(Account account) =>
|
2022-01-16 22:38:09 +01:00
|
|
|
"remote.php/dav/files/${account.homeDir}";
|
2021-04-10 06:28:12 +02:00
|
|
|
|
2021-08-01 22:46:16 +02:00
|
|
|
String getTrashbinPath(Account account) =>
|
2022-01-16 22:38:09 +01:00
|
|
|
"remote.php/dav/trashbin/${account.homeDir}/trash";
|
2021-08-01 22:46:16 +02:00
|
|
|
|
2021-09-08 12:44:14 +02:00
|
|
|
/// Return the face image URL. See [getFacePreviewUrlRelative]
|
|
|
|
String getFacePreviewUrl(
|
|
|
|
Account account,
|
2021-09-10 19:10:26 +02:00
|
|
|
int faceId, {
|
2021-09-08 12:44:14 +02:00
|
|
|
required int size,
|
|
|
|
}) {
|
|
|
|
return "${account.url}/"
|
2021-09-10 19:10:26 +02:00
|
|
|
"${getFacePreviewUrlRelative(account, faceId, size: size)}";
|
2021-09-08 12:44:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the relative URL of the face image
|
|
|
|
String getFacePreviewUrlRelative(
|
|
|
|
Account account,
|
2021-09-10 19:10:26 +02:00
|
|
|
int faceId, {
|
2021-09-08 12:44:14 +02:00
|
|
|
required int size,
|
|
|
|
}) {
|
2021-09-10 19:10:26 +02:00
|
|
|
return "index.php/apps/facerecognition/face/$faceId/thumb/$size";
|
2021-09-08 12:44:14 +02:00
|
|
|
}
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
/// Query the app password for [account]
|
|
|
|
Future<String> exchangePassword(Account account) async {
|
|
|
|
final response = await Api(account).request(
|
|
|
|
"GET",
|
|
|
|
"ocs/v2.php/core/getapppassword",
|
|
|
|
header: {
|
|
|
|
"OCS-APIRequest": "true",
|
|
|
|
},
|
|
|
|
);
|
|
|
|
if (response.isGood) {
|
2021-05-06 06:57:20 +02:00
|
|
|
try {
|
|
|
|
final appPwdRegex = RegExp(r"<apppassword>(.*)</apppassword>");
|
|
|
|
final appPwdMatch = appPwdRegex.firstMatch(response.body);
|
2021-07-23 22:05:57 +02:00
|
|
|
return appPwdMatch!.group(1)!;
|
2021-05-06 06:57:20 +02:00
|
|
|
} catch (_) {
|
|
|
|
// this happens when the address is not the base URL and so Nextcloud
|
|
|
|
// returned the login page
|
|
|
|
throw InvalidBaseUrlException();
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
} else if (response.statusCode == 403) {
|
|
|
|
// If the client is authenticated with an app password a 403 will be
|
|
|
|
// returned
|
|
|
|
_log.info("[exchangePassword] Already an app password");
|
|
|
|
return account.password;
|
|
|
|
} else {
|
|
|
|
_log.severe(
|
|
|
|
"[exchangePassword] Failed while requesting app password: $response");
|
2021-05-07 21:22:21 +02:00
|
|
|
throw ApiException(
|
|
|
|
response: response,
|
|
|
|
message: "Failed communicating with server: ${response.statusCode}");
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final _log = Logger("api.api_util");
|