mirror of
https://github.com/lollipopkit/flutter_server_box.git
synced 2025-12-17 07:14:28 +01:00
new: webdav sync
This commit is contained in:
@@ -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';
|
||||||
96
lib/core/utils/sync/webdav.dart
Normal file
96
lib/core/utils/sync/webdav.dart
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
13
lib/data/model/app/remote_storage.dart
Normal file
13
lib/data/model/app/remote_storage.dart
Normal 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);
|
||||||
|
}
|
||||||
@@ -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------
|
||||||
|
|||||||
@@ -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';
|
||||||
|
|||||||
@@ -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();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
Reference in New Issue
Block a user