new: webdav sync

This commit is contained in:
lollipopkit
2023-12-04 13:39:13 +08:00
parent 3524d92013
commit 2dc86a9da2
9 changed files with 385 additions and 84 deletions

View File

@@ -7,8 +7,8 @@ import 'package:toolbox/data/model/app/backup.dart';
import 'package:toolbox/data/model/app/sync.dart'; import 'package:toolbox/data/model/app/sync.dart';
import 'package:toolbox/data/res/logger.dart'; import 'package:toolbox/data/res/logger.dart';
import '../../data/model/app/error.dart'; import '../../../data/model/app/error.dart';
import '../../data/res/path.dart'; import '../../../data/res/path.dart';
abstract final class ICloud { abstract final class ICloud {
static const _containerId = 'iCloud.tech.lolli.serverbox'; static const _containerId = 'iCloud.tech.lolli.serverbox';

View File

@@ -0,0 +1,96 @@
import 'dart:io';
import 'package:flutter/foundation.dart';
import 'package:toolbox/data/model/app/backup.dart';
import 'package:toolbox/data/model/app/error.dart';
import 'package:toolbox/data/res/logger.dart';
import 'package:toolbox/data/res/path.dart';
import 'package:toolbox/data/res/store.dart';
// ignore: implementation_imports
import 'package:webdav_client/src/client.dart';
abstract final class Webdav {
static var _client = WebdavClient(
url: Stores.setting.webdavUrl.fetch(),
user: Stores.setting.webdavUser.fetch(),
pwd: Stores.setting.webdavPwd.fetch(),
);
static Future<WebdavErr?> upload({
required String relativePath,
String? localPath,
}) async {
try {
await _client.writeFile(
localPath ?? '${await Paths.doc}/$relativePath',
relativePath,
);
} catch (e, s) {
Loggers.app.warning('Webdav upload failed', e, s);
return WebdavErr(type: WebdavErrType.generic, message: '$e');
}
return null;
}
static Future<WebdavErr?> delete(String relativePath) async {
try {
await _client.remove(relativePath);
} catch (e, s) {
Loggers.app.warning('Webdav delete failed', e, s);
return WebdavErr(type: WebdavErrType.generic, message: '$e');
}
return null;
}
static Future<WebdavErr?> download({
required String relativePath,
String? localPath,
}) async {
try {
await _client.readFile(
relativePath,
localPath ?? '${await Paths.doc}/$relativePath',
);
} catch (e, s) {
Loggers.app.warning('Webdav download failed', e, s);
return WebdavErr(type: WebdavErrType.generic, message: '$e');
}
return null;
}
static void changeClient(String url, String user, String pwd) {
_client = WebdavClient(url: url, user: user, pwd: pwd);
}
static Future<void> sync() async {
try {
final result = await download(relativePath: Paths.bakName);
if (result != null) {
Loggers.app.warning('Download backup failed: $result');
return;
}
} catch (e, s) {
Loggers.app.warning('Download backup failed', e, s);
}
final dlFile = await File(await Paths.bak).readAsString();
final dlBak = await compute(Backup.fromJsonString, dlFile);
final restore = await dlBak.restore();
switch (restore) {
case true:
Loggers.app.info('Restore from iCloud (${dlBak.lastModTime}) success');
break;
case false:
await Backup.backup();
final uploadResult = await upload(relativePath: Paths.bakName);
if (uploadResult != null) {
Loggers.app.warning('Upload iCloud backup failed: $uploadResult');
} else {
Loggers.app.info('Upload iCloud backup success');
}
break;
case null:
Loggers.app.info('Skip iCloud sync');
break;
}
}
}

View File

@@ -4,7 +4,9 @@ enum ErrFrom {
docker, docker,
sftp, sftp,
ssh, ssh,
status; status,
icloud,
webdav,;
} }
abstract class Err<T> { abstract class Err<T> {
@@ -61,10 +63,25 @@ enum ICloudErrType {
class ICloudErr extends Err<ICloudErrType> { class ICloudErr extends Err<ICloudErrType> {
ICloudErr({required ICloudErrType type, String? message}) ICloudErr({required ICloudErrType type, String? message})
: super(from: ErrFrom.docker, type: type, message: message); : super(from: ErrFrom.icloud, type: type, message: message);
@override @override
String toString() { String toString() {
return 'ICloudErr<$type>: $message'; return 'ICloudErr<$type>: $message';
} }
} }
enum WebdavErrType {
generic,
notFound,;
}
class WebdavErr extends Err<WebdavErrType> {
WebdavErr({required WebdavErrType type, String? message})
: super(from: ErrFrom.webdav, type: type, message: message);
@override
String toString() {
return 'WebdavErr<$type>: $message';
}
}

View File

@@ -0,0 +1,13 @@
abstract class RemoteStorage {
Future<Error> upload({
required String relativePath,
String? localPath
});
Future<Error> download({
required String relativePath,
String? localPath
});
Future<Error> delete(String relativePath);
}

View File

@@ -203,6 +203,12 @@ class SettingStore extends PersistentStore {
late final serverFuncBtnsDisplayName = late final serverFuncBtnsDisplayName =
property('serverFuncBtnsDisplayName', false); property('serverFuncBtnsDisplayName', false);
/// Webdav sync
late final webdavSync = property('webdavSync', false);
late final webdavUrl = property('webdavUrl', '');
late final webdavUser = property('webdavUser', '');
late final webdavPwd = property('webdavPwd', '');
// Never show these settings for users // Never show these settings for users
// //
// ------BEGIN------ // ------BEGIN------

View File

@@ -9,7 +9,7 @@ import 'package:macos_window_utils/window_manipulator.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'package:shared_preferences/shared_preferences.dart'; import 'package:shared_preferences/shared_preferences.dart';
import 'package:toolbox/core/channel/bg_run.dart'; import 'package:toolbox/core/channel/bg_run.dart';
import 'package:toolbox/core/utils/icloud.dart'; import 'package:toolbox/core/utils/sync/icloud.dart';
import 'package:toolbox/core/utils/platform/base.dart'; import 'package:toolbox/core/utils/platform/base.dart';
import 'package:toolbox/data/res/logger.dart'; import 'package:toolbox/data/res/logger.dart';
import 'package:toolbox/data/res/provider.dart'; import 'package:toolbox/data/res/provider.dart';

View File

@@ -1,4 +1,3 @@
import 'dart:async';
import 'dart:io'; import 'dart:io';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
@@ -7,17 +6,19 @@ import 'package:toolbox/core/extension/context/common.dart';
import 'package:toolbox/core/extension/context/dialog.dart'; import 'package:toolbox/core/extension/context/dialog.dart';
import 'package:toolbox/core/extension/context/locale.dart'; import 'package:toolbox/core/extension/context/locale.dart';
import 'package:toolbox/core/extension/context/snackbar.dart'; import 'package:toolbox/core/extension/context/snackbar.dart';
import 'package:toolbox/core/persistant_store.dart'; import 'package:toolbox/core/utils/sync/icloud.dart';
import 'package:toolbox/core/utils/icloud.dart';
import 'package:toolbox/core/utils/platform/base.dart'; import 'package:toolbox/core/utils/platform/base.dart';
import 'package:toolbox/core/utils/share.dart'; import 'package:toolbox/core/utils/share.dart';
import 'package:toolbox/core/utils/sync/webdav.dart';
import 'package:toolbox/data/model/app/backup.dart'; import 'package:toolbox/data/model/app/backup.dart';
import 'package:toolbox/data/res/logger.dart'; import 'package:toolbox/data/res/logger.dart';
import 'package:toolbox/data/res/path.dart';
import 'package:toolbox/data/res/provider.dart'; import 'package:toolbox/data/res/provider.dart';
import 'package:toolbox/data/res/rebuild.dart'; import 'package:toolbox/data/res/rebuild.dart';
import 'package:toolbox/data/res/store.dart'; import 'package:toolbox/data/res/store.dart';
import 'package:toolbox/view/widget/expand_tile.dart'; import 'package:toolbox/view/widget/expand_tile.dart';
import 'package:toolbox/view/widget/cardx.dart'; import 'package:toolbox/view/widget/cardx.dart';
import 'package:toolbox/view/widget/input_field.dart';
import 'package:toolbox/view/widget/store_switch.dart'; import 'package:toolbox/view/widget/store_switch.dart';
import 'package:toolbox/view/widget/value_notifier.dart'; import 'package:toolbox/view/widget/value_notifier.dart';
@@ -29,6 +30,7 @@ class BackupPage extends StatelessWidget {
BackupPage({Key? key}) : super(key: key); BackupPage({Key? key}) : super(key: key);
final icloudLoading = ValueNotifier(false); final icloudLoading = ValueNotifier(false);
final webdavLoading = ValueNotifier(false);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@@ -45,6 +47,7 @@ class BackupPage extends StatelessWidget {
padding: const EdgeInsets.all(17), padding: const EdgeInsets.all(17),
children: [ children: [
if (isMacOS || isIOS) _buildIcloud(context), if (isMacOS || isIOS) _buildIcloud(context),
_buildWebdav(context),
_buildFile(context), _buildFile(context),
], ],
); );
@@ -53,6 +56,7 @@ class BackupPage extends StatelessWidget {
Widget _buildFile(BuildContext context) { Widget _buildFile(BuildContext context) {
return CardX( return CardX(
ExpandTile( ExpandTile(
leading: const Icon(Icons.file_open, size: 19),
title: Text(l10n.files), title: Text(l10n.files),
initiallyExpanded: true, initiallyExpanded: true,
children: [ children: [
@@ -63,12 +67,73 @@ class BackupPage extends StatelessWidget {
l10n.backupTip, l10n.backupTip,
style: UIs.textGrey, style: UIs.textGrey,
), ),
onTap: _onBackup, onTap: () async {
final path = await Backup.backup();
/// Issue #188
if (isWindows) {
await Shares.text(await File(path).readAsString());
} else {
await Shares.files([path]);
}
},
), ),
ListTile( ListTile(
trailing: const Icon(Icons.restore), trailing: const Icon(Icons.restore),
title: Text(l10n.restore), title: Text(l10n.restore),
onTap: () => _onRestore(context), onTap: () async {
final path = await pickOneFile();
if (path == null) return;
final file = File(path);
if (!await file.exists()) {
context.showSnackBar(l10n.fileNotExist(path));
return;
}
final text = await file.readAsString();
if (text.isEmpty) {
context.showSnackBar(l10n.fieldMustNotEmpty);
return;
}
try {
context.showLoadingDialog();
final backup =
await compute(Backup.fromJsonString, text.trim());
if (backupFormatVersion != backup.version) {
context.showSnackBar(l10n.backupVersionNotMatch);
return;
}
await context.showRoundDialog(
title: Text(l10n.restore),
child: Text(l10n.askContinue(
'${l10n.restore} ${l10n.backup}(${backup.date})',
)),
actions: [
TextButton(
onPressed: () => context.pop(),
child: Text(l10n.cancel),
),
TextButton(
onPressed: () async {
await backup.restore(force: true);
Pros.reload();
context.pop();
RebuildNodes.app.rebuild();
},
child: Text(l10n.ok),
),
],
);
} catch (e, trace) {
Loggers.app.warning('Import backup failed', e, trace);
context.showSnackBar(e.toString());
} finally {
context.pop();
}
},
), ),
], ],
), ),
@@ -78,23 +143,19 @@ class BackupPage extends StatelessWidget {
Widget _buildIcloud(BuildContext context) { Widget _buildIcloud(BuildContext context) {
return CardX( return CardX(
ExpandTile( ExpandTile(
leading: const Icon(Icons.cloud, size: 19),
title: const Text('iCloud'), title: const Text('iCloud'),
initiallyExpanded: true, initiallyExpanded: true,
subtitle: Text(
l10n.syncTip,
style: UIs.textGrey,
),
children: [ children: [
ListTile( ListTile(
title: Text(l10n.auto), title: Text(l10n.auto),
subtitle: const Text(
'Unavailable, please wait for optimization :)',
style: UIs.textGrey,
),
trailing: StoreSwitch( trailing: StoreSwitch(
prop: Stores.setting.icloudSync, prop: Stores.setting.icloudSync,
func: (val) async { func: (val) async {
if (val) { if (val) {
icloudLoading.value = true;
await ICloud.sync();
icloudLoading.value = false;
} }
}, },
), ),
@@ -113,10 +174,26 @@ class BackupPage extends StatelessWidget {
TextButton( TextButton(
onPressed: () async { onPressed: () async {
icloudLoading.value = true; icloudLoading.value = true;
final files = await PersistentStore.getFileNames(); try {
for (final file in files) { final result = await ICloud.download(
await ICloud.download(relativePath: file); relativePath: Paths.bakName,
);
if (result != null) {
Loggers.app
.warning('Download backup failed: $result');
return;
}
} catch (e, s) {
Loggers.app.warning('Download backup failed', e, s);
context.showSnackBar(e.toString());
icloudLoading.value = false;
return;
} }
final dlFile =
await File(await Paths.bak).readAsString();
final dlBak =
await compute(Backup.fromJsonString, dlFile);
await dlBak.restore(force: true);
icloudLoading.value = false; icloudLoading.value = false;
}, },
child: Text(l10n.download), child: Text(l10n.download),
@@ -125,9 +202,14 @@ class BackupPage extends StatelessWidget {
TextButton( TextButton(
onPressed: () async { onPressed: () async {
icloudLoading.value = true; icloudLoading.value = true;
final files = await PersistentStore.getFileNames(); await Backup.backup();
for (final file in files) { final uploadResult =
await ICloud.upload(relativePath: file); await ICloud.upload(relativePath: Paths.bakName);
if (uploadResult != null) {
Loggers.app.warning(
'Upload iCloud backup failed: $uploadResult');
} else {
Loggers.app.info('Upload iCloud backup success');
} }
icloudLoading.value = false; icloudLoading.value = false;
}, },
@@ -143,68 +225,142 @@ class BackupPage extends StatelessWidget {
); );
} }
Future<void> _onBackup() async { Widget _buildWebdav(BuildContext context) {
final path = await Backup.backup(); return CardX(
ExpandTile(
/// Issue #188 leading: const Icon(Icons.storage, size: 19),
if (isWindows) { title: const Text('WebDAV'),
await Shares.text(await File(path).readAsString()); initiallyExpanded: !(isIOS || isMacOS),
} else { children: [
await Shares.files([path]); ListTile(
} title: Text(l10n.setting),
} trailing: const Icon(Icons.settings),
onTap: () async {
Future<void> _onRestore(BuildContext context) async { final urlCtrl = TextEditingController(
final path = await pickOneFile(); text: Stores.setting.webdavUrl.fetch(),
if (path == null) return; );
final userCtrl = TextEditingController(
final file = File(path); text: Stores.setting.webdavUser.fetch(),
if (!await file.exists()) { );
context.showSnackBar(l10n.fileNotExist(path)); final pwdCtrl = TextEditingController(
return; text: Stores.setting.webdavPwd.fetch(),
} );
final result = await context.showRoundDialog<bool>(
final text = await file.readAsString(); title: const Text('WebDAV'),
if (text.isEmpty) { child: Column(
context.showSnackBar(l10n.fieldMustNotEmpty); mainAxisSize: MainAxisSize.min,
return; children: [
} Input(
label: 'url',
try { controller: urlCtrl,
context.showLoadingDialog(); ),
final backup = await compute(Backup.fromJsonString, text.trim()); Input(
if (backupFormatVersion != backup.version) { label: l10n.user,
context.showSnackBar(l10n.backupVersionNotMatch); controller: userCtrl,
return; ),
} Input(
label: l10n.pwd,
await context.showRoundDialog( controller: pwdCtrl,
title: Text(l10n.restore), ),
child: Text(l10n.askContinue( ],
'${l10n.restore} ${l10n.backup}(${backup.date})', ),
)), actions: [
actions: [ TextButton(
TextButton( onPressed: () {
onPressed: () => context.pop(), context.pop(true);
child: Text(l10n.cancel), },
), child: Text(l10n.ok),
TextButton( ),
onPressed: () async { ],
/// TODO: add checkbox for not force restore );
await backup.restore(force: true); if (result == true) {
Pros.reload(); Webdav.changeClient(
context.pop(); urlCtrl.text,
RebuildNodes.app.rebuild(); userCtrl.text,
pwdCtrl.text,
);
Stores.setting.webdavUrl.put(urlCtrl.text);
Stores.setting.webdavUser.put(userCtrl.text);
Stores.setting.webdavPwd.put(pwdCtrl.text);
}
}, },
child: Text(l10n.ok), ),
ListTile(
title: Text(l10n.auto),
trailing: StoreSwitch(
prop: Stores.setting.webdavSync,
func: (val) async {
if (val) {
webdavLoading.value = true;
await Webdav.sync();
webdavLoading.value = false;
}
},
),
),
ListTile(
title: Text(l10n.manual),
trailing: ValueBuilder(
listenable: webdavLoading,
build: () {
if (webdavLoading.value) {
return UIs.centerSizedLoadingSmall;
}
return Row(
mainAxisSize: MainAxisSize.min,
children: [
TextButton(
onPressed: () async {
webdavLoading.value = true;
try {
final result = await Webdav.download(
relativePath: Paths.bakName,
);
if (result != null) {
Loggers.app.warning(
'Download webdav backup failed: $result');
return;
}
} catch (e, s) {
Loggers.app
.warning('Download webdav backup failed', e, s);
context.showSnackBar(e.toString());
webdavLoading.value = false;
return;
}
final dlFile =
await File(await Paths.bak).readAsString();
final dlBak =
await compute(Backup.fromJsonString, dlFile);
await dlBak.restore(force: true);
webdavLoading.value = false;
},
child: Text(l10n.download),
),
UIs.width7,
TextButton(
onPressed: () async {
webdavLoading.value = true;
await Backup.backup();
final uploadResult =
await Webdav.upload(relativePath: Paths.bakName);
if (uploadResult != null) {
Loggers.app.warning(
'Upload webdav backup failed: $uploadResult');
} else {
Loggers.app.info('Upload webdav backup success');
}
webdavLoading.value = false;
},
child: Text(l10n.upload),
),
],
);
},
),
), ),
], ],
); ),
} catch (e, trace) { );
Loggers.app.warning('Import backup failed', e, trace);
context.showSnackBar(e.toString());
} finally {
context.pop();
}
} }
} }

View File

@@ -1164,6 +1164,15 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.4.0" version: "2.4.0"
webdav_client:
dependency: "direct main"
description:
path: "."
ref: main
resolved-ref: "233b3ebaa01b7bb35a414c9cfd5e2933b3a008ba"
url: "https://github.com/lollipopkit/webdav_client"
source: git
version: "1.2.1"
win32: win32:
dependency: transitive dependency: transitive
description: description:

View File

@@ -62,6 +62,10 @@ dependencies:
#flutter_secure_storage: ^9.0.0 #flutter_secure_storage: ^9.0.0
xml: ^6.4.2 xml: ^6.4.2
flutter_rfb: ^0.6.2 flutter_rfb: ^0.6.2
webdav_client:
git:
ref: main
url: https://github.com/lollipopkit/webdav_client
dev_dependencies: dev_dependencies:
flutter_native_splash: ^2.1.6 flutter_native_splash: ^2.1.6