cake_wallet/cw_zano/lib/zano_wallet_service.dart

258 lines
9.4 KiB
Dart
Raw Normal View History

2023-12-14 04:51:16 +00:00
import 'dart:convert';
2023-10-02 14:17:35 +00:00
import 'dart:io';
2023-10-02 14:17:35 +00:00
import 'package:collection/collection.dart';
2023-12-14 04:51:16 +00:00
import 'package:cw_core/crypto_currency.dart';
2023-11-17 17:40:23 +00:00
import 'package:cw_core/node.dart';
import 'package:cw_core/pathForWallet.dart';
2023-10-02 14:17:35 +00:00
import 'package:cw_core/wallet_base.dart';
import 'package:cw_core/wallet_credentials.dart';
import 'package:cw_core/wallet_info.dart';
import 'package:cw_core/wallet_service.dart';
2023-10-02 14:17:35 +00:00
import 'package:cw_core/wallet_type.dart';
2024-03-06 06:48:59 +00:00
import 'package:cw_zano/api/api_calls.dart' as calls;
import 'package:cw_zano/api/api_calls.dart';
2023-12-16 12:19:11 +00:00
import 'package:cw_zano/api/consts.dart';
import 'package:cw_zano/api/exceptions/already_exists_exception.dart';
2023-12-16 15:00:22 +00:00
import 'package:cw_zano/api/exceptions/create_wallet_exception.dart';
2023-12-16 12:19:11 +00:00
import 'package:cw_zano/api/exceptions/restore_from_seed_exception.dart';
import 'package:cw_zano/api/exceptions/wrong_seed_exception.dart';
import 'package:cw_zano/api/model/create_wallet_result.dart';
2024-03-15 12:42:27 +00:00
import 'package:cw_zano/zano_asset.dart';
import 'package:cw_zano/zano_balance.dart';
import 'package:cw_zano/zano_wallet.dart';
import 'package:hive/hive.dart';
2023-12-14 04:51:16 +00:00
import 'package:mobx/mobx.dart';
2023-10-02 14:17:35 +00:00
class ZanoNewWalletCredentials extends WalletCredentials {
ZanoNewWalletCredentials({required String name, String? password}) : super(name: name, password: password);
2023-10-02 14:17:35 +00:00
}
class ZanoRestoreWalletFromSeedCredentials extends WalletCredentials {
ZanoRestoreWalletFromSeedCredentials({required String name, required String password, required int height, required this.mnemonic})
2023-10-02 14:17:35 +00:00
: super(name: name, password: password, height: height);
final String mnemonic;
}
class ZanoRestoreWalletFromKeysCredentials extends WalletCredentials {
ZanoRestoreWalletFromKeysCredentials(
2024-03-15 12:42:27 +00:00
{required String name,
required String password,
required this.language,
required this.address,
required this.viewKey,
required this.spendKey,
required int height})
2023-10-02 14:17:35 +00:00
: super(name: name, password: password, height: height);
final String language;
final String address;
final String viewKey;
final String spendKey;
}
class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRestoreWalletFromSeedCredentials, ZanoRestoreWalletFromKeysCredentials> {
2023-10-02 14:17:35 +00:00
ZanoWalletService(this.walletInfoSource);
final Box<WalletInfo> walletInfoSource;
static bool walletFilesExist(String path) => !File(path).existsSync() && !File('$path.keys').existsSync();
2023-10-02 14:17:35 +00:00
2023-11-17 17:40:23 +00:00
int hWallet = 0;
2023-10-02 14:17:35 +00:00
@override
WalletType getType() => WalletType.zano;
@override
Future<ZanoWallet> create(WalletCredentials credentials, {bool? isTestnet}) async {
print('zanowallet service create isTestnet $isTestnet'); // TODO: remove
2023-10-02 14:17:35 +00:00
try {
2023-12-14 04:51:16 +00:00
final wallet = ZanoWallet(credentials.walletInfo!);
await wallet.connectToNode(node: Node());
2023-10-02 14:17:35 +00:00
final path = await pathForWallet(name: credentials.name, type: getType());
final result = ApiCalls.createWallet(language: '', path: path, password: credentials.password!);
2023-12-14 04:51:16 +00:00
final map = json.decode(result) as Map<String, dynamic>;
_checkForCreateWalletError(map);
2023-12-16 15:00:22 +00:00
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
_parseCreateWalletResult(createWalletResult, wallet);
2024-03-06 06:48:59 +00:00
await wallet.store();
2023-12-16 15:00:22 +00:00
await wallet.init(createWalletResult.wi.address);
wallet.addInitialAssets();
2023-10-02 14:17:35 +00:00
return wallet;
} catch (e) {
// TODO: Implement Exception for wallet list service.
print('ZanoWalletsManager Error: ${e.toString()}');
rethrow;
}
}
@override
Future<bool> isWalletExit(String name) async {
try {
final path = await pathForWallet(name: name, type: getType());
2024-03-06 06:48:59 +00:00
return ApiCalls.isWalletExist(path: path);
2023-10-02 14:17:35 +00:00
} catch (e) {
// TODO: Implement Exception for wallet list service.
print('ZanoWalletsManager Error: $e');
rethrow;
}
}
@override
Future<ZanoWallet> openWallet(String name, String password) async {
try {
final path = await pathForWallet(name: name, type: getType());
if (walletFilesExist(path)) {
await repairOldAndroidWallet(name);
}
final walletInfo = walletInfoSource.values.firstWhereOrNull((info) => info.id == WalletBase.idFor(name, getType()))!;
2023-12-14 04:51:16 +00:00
final wallet = ZanoWallet(walletInfo);
await wallet.connectToNode(node: Node());
final result = wallet.loadWallet(path, password);
print('load wallet result $result');
2023-12-14 04:51:16 +00:00
final map = json.decode(result) as Map<String, dynamic>;
_checkForCreateWalletError(map);
2023-12-16 15:00:22 +00:00
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
_parseCreateWalletResult(createWalletResult, wallet);
2024-03-06 06:48:59 +00:00
await wallet.store();
2023-12-16 15:00:22 +00:00
await wallet.init(createWalletResult.wi.address);
2023-10-02 14:17:35 +00:00
return wallet;
} catch (e) {
rethrow;
// TODO: uncomment after merge
//await restoreWalletFilesFromBackup(name);
}
}
void _checkForCreateWalletError(Map<String, dynamic> map) {
if (map['error'] != null) {
final code = map['error']!['code'] ?? '';
final message = map['error']!['message'] ?? '';
throw CreateWalletException('Error creating/loading wallet $code $message');
}
2024-03-15 12:42:27 +00:00
if (map['result'] == null) {
throw CreateWalletException('Error creating/loading wallet, empty response');
2023-10-02 14:17:35 +00:00
}
}
2023-12-14 04:51:16 +00:00
void _parseCreateWalletResult(CreateWalletResult result, ZanoWallet wallet) {
hWallet = result.walletId;
wallet.hWallet = hWallet;
wallet.walletAddresses.address = result.wi.address;
2024-03-15 12:42:27 +00:00
for (final item in result.wi.balances) {
if (item.assetInfo.ticker == 'ZANO') {
wallet.balance[CryptoCurrency.zano] = ZanoBalance(total: item.total, unlocked: item.unlocked);
} else {
for (final asset in wallet.balance.keys) {
if (asset is ZanoAsset && asset.assetId == item.assetInfo.assetId) {
wallet.balance[asset] = ZanoBalance(total: item.total, unlocked: item.unlocked);
}
}
}
}
2023-12-14 04:51:16 +00:00
if (result.recentHistory.history != null) {
wallet.history = result.recentHistory.history!;
}
}
2023-10-02 14:17:35 +00:00
@override
Future<void> remove(String wallet) async {
final path = await pathForWalletDir(name: wallet, type: getType());
final file = Directory(path);
final isExist = file.existsSync();
if (isExist) {
await file.delete(recursive: true);
}
final walletInfo = walletInfoSource.values.firstWhere((info) => info.id == WalletBase.idFor(wallet, getType()));
2023-10-02 14:17:35 +00:00
await walletInfoSource.delete(walletInfo.key);
}
@override
2023-12-14 04:51:16 +00:00
Future<void> rename(String currentName, String password, String newName) async {
final currentWalletInfo = walletInfoSource.values.firstWhere((info) => info.id == WalletBase.idFor(currentName, getType()));
2023-12-14 04:51:16 +00:00
final currentWallet = ZanoWallet(currentWalletInfo);
2023-10-02 14:17:35 +00:00
await currentWallet.renameWalletFiles(newName);
final newWalletInfo = currentWalletInfo;
newWalletInfo.id = WalletBase.idFor(newName, getType());
newWalletInfo.name = newName;
await walletInfoSource.put(currentWalletInfo.key, newWalletInfo);
}
@override
Future<ZanoWallet> restoreFromKeys(ZanoRestoreWalletFromKeysCredentials credentials, {bool? isTestnet}) async {
throw UnimplementedError('Restore from keys not implemented');
2023-10-02 14:17:35 +00:00
}
@override
Future<ZanoWallet> restoreFromSeed(ZanoRestoreWalletFromSeedCredentials credentials, {bool? isTestnet}) async {
2023-10-02 14:17:35 +00:00
try {
2023-12-14 04:51:16 +00:00
final wallet = ZanoWallet(credentials.walletInfo!);
await wallet.connectToNode(node: Node());
2023-10-02 14:17:35 +00:00
final path = await pathForWallet(name: credentials.name, type: getType());
2024-03-06 06:48:59 +00:00
final result = ApiCalls.restoreWalletFromSeed(path: path, password: credentials.password!, seed: credentials.mnemonic);
2023-12-16 12:19:11 +00:00
final map = json.decode(result) as Map<String, dynamic>;
2023-12-14 04:51:16 +00:00
if (map['result'] != null) {
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
2023-12-14 04:51:16 +00:00
_parseCreateWalletResult(createWalletResult, wallet);
2024-03-06 06:48:59 +00:00
await wallet.store();
2023-12-16 15:00:22 +00:00
await wallet.init(createWalletResult.wi.address);
wallet.addInitialAssets();
2023-12-16 15:00:22 +00:00
return wallet;
2023-12-16 12:19:11 +00:00
} else if (map['error'] != null) {
final code = map['error']['code'] as String;
final message = map['error']['message'] as String;
if (code == Consts.errorWrongSeed) {
throw WrongSeedException(message);
} else if (code == Consts.errorAlreadyExists) {
throw AlreadyExistsException(message);
}
throw RestoreFromSeedException(code, message);
2023-12-14 04:51:16 +00:00
}
throw RestoreFromSeedException('', '');
2023-10-02 14:17:35 +00:00
} catch (e) {
// TODO: Implement Exception for wallet list service.
print('ZanoWalletsManager Error: $e');
rethrow;
}
}
Future<void> repairOldAndroidWallet(String name) async {
try {
if (!Platform.isAndroid) {
return;
}
2023-12-14 04:51:16 +00:00
final oldAndroidWalletDirPath = await outdatedAndroidPathForWalletDir(name: name);
2023-10-02 14:17:35 +00:00
final dir = Directory(oldAndroidWalletDirPath);
if (!dir.existsSync()) {
return;
}
2023-12-14 04:51:16 +00:00
final newWalletDirPath = await pathForWalletDir(name: name, type: getType());
2023-10-02 14:17:35 +00:00
dir.listSync().forEach((f) {
final file = File(f.path);
final name = f.path.split('/').last;
final newPath = newWalletDirPath + '/$name';
final newFile = File(newPath);
if (!newFile.existsSync()) {
newFile.createSync();
}
newFile.writeAsBytesSync(file.readAsBytesSync());
});
} catch (e) {
print(e.toString());
}
}
}