open, create, restore wallet refactoring; whitelists

This commit is contained in:
leo 2024-04-03 15:14:53 +00:00
parent 9d9fe4a5a6
commit 42731fcdcb
24 changed files with 394 additions and 478 deletions

View file

@ -1,6 +1,7 @@
import 'dart:ffi'; import 'dart:ffi';
import 'package:cw_zano/api/structs/utf8_box.dart'; import 'package:cw_zano/api/utf8.dart';
import 'package:cw_zano/api/utf8_box.dart';
import 'package:cw_zano/api/zano_api.dart'; import 'package:cw_zano/api/zano_api.dart';
import 'package:ffi/ffi.dart'; import 'package:ffi/ffi.dart';
@ -55,7 +56,7 @@ typedef _stringFunction = Pointer<Utf8> Function();
class ApiCalls { class ApiCalls {
static String _convertUTF8ToString({required Pointer<Utf8> pointer}) { static String _convertUTF8ToString({required Pointer<Utf8> pointer}) {
final str = pointer.toDartString(); final str = pointer.toDartStringAllowingMalformed();
calloc.free(pointer); calloc.free(pointer);
return str; return str;
} }

View file

@ -1,4 +1,5 @@
class Consts { class Consts {
static const errorWrongSeed = 'WRONG_SEED'; static const errorWrongSeed = 'WRONG_SEED';
static const errorAlreadyExists = 'ALREADY_EXISTS'; static const errorAlreadyExists = 'ALREADY_EXISTS';
static const errorWalletWrongId = 'WALLET_WRONG_ID';
} }

View file

@ -1,6 +0,0 @@
import 'package:cw_zano/api/consts.dart';
import 'package:cw_zano/api/exceptions/api_exception.dart';
class AlreadyExistsException extends ApiException {
AlreadyExistsException(String message): super(Consts.errorAlreadyExists, message);
}

View file

@ -1,9 +0,0 @@
class ApiException implements Exception {
final String code;
final String message;
ApiException(this.code, this.message);
@override
String toString() => '${this.runtimeType}(code: $code, message: $message)';
}

View file

@ -1,7 +0,0 @@
class CreateWalletException implements Exception {
final String message;
CreateWalletException(this.message): super();
@override
String toString() => '${this.runtimeType}(message: $message)';
}

View file

@ -1,5 +0,0 @@
import 'package:cw_zano/api/exceptions/api_exception.dart';
class RestoreFromSeedException extends ApiException {
RestoreFromSeedException(String code, String message): super(code, message);
}

View file

@ -1,5 +0,0 @@
import 'package:cw_zano/api/exceptions/api_exception.dart';
class TransferException extends ApiException {
TransferException(String code, String message): super(code, message);
}

View file

@ -1,5 +0,0 @@
class WalletRestoreFromKeysException implements Exception {
WalletRestoreFromKeysException({required this.message});
final String message;
}

View file

@ -1,6 +0,0 @@
import 'package:cw_zano/api/consts.dart';
import 'package:cw_zano/api/exceptions/api_exception.dart';
class WrongSeedException extends ApiException {
WrongSeedException(String message): super(Consts.errorWrongSeed, message);
}

View file

@ -1,5 +1,3 @@
import 'package:cw_core/amount_converter.dart';
import 'package:cw_core/crypto_currency.dart';
import 'package:cw_zano/model/zano_asset.dart'; import 'package:cw_zano/model/zano_asset.dart';
class Balance { class Balance {
@ -17,7 +15,7 @@ class Balance {
required this.unlocked}); required this.unlocked});
@override @override
String toString() => '$assetInfo: ${AmountConverter.amountIntToString(CryptoCurrency.zano, total)}/${AmountConverter.amountIntToString(CryptoCurrency.zano, unlocked)}'; String toString() => '$assetInfo: $total/$unlocked';
factory Balance.fromJson(Map<String, dynamic> json) => Balance( factory Balance.fromJson(Map<String, dynamic> json) => Balance(
assetInfo: assetInfo:

25
cw_zano/lib/api/utf8.dart Normal file
View file

@ -0,0 +1,25 @@
import 'dart:convert';
import 'dart:ffi';
import 'package:ffi/ffi.dart';
extension Utf8Pointer on Pointer<Utf8> {
String toDartStringAllowingMalformed({int? length}) {
//_ensureNotNullptr('toDartString');
final codeUnits = cast<Uint8>();
if (length != null) {
RangeError.checkNotNegative(length, 'length');
} else {
length = _length(codeUnits);
}
return utf8.decode(codeUnits.asTypedList(length), allowMalformed: true);
}
static int _length(Pointer<Uint8> codeUnits) {
var length = 0;
while (codeUnits[length] != 0) {
length++;
}
return length;
}
}

View file

@ -1,33 +0,0 @@
import 'package:cw_core/crypto_currency.dart';
import 'package:cw_zano/model/zano_asset.dart';
class DefaultZanoAssets {
final List<ZanoAsset> _defaultAssets = [
ZanoAsset(
decimalPoint: 12,
fullName: 'Confidential token',
assetId: 'cc4e69455e63f4a581257382191de6856c2156630b3fba0db4bdd73ffcfb36b6',
owner: '32911fabcf90b9731a152d2a3a75fcbb0a46c78e2f502678bae44c3d6823b4ce',
ticker: 'CT',
enabled: false,
),
ZanoAsset(
decimalPoint: 12,
fullName: '새로운경제',
assetId: 'bb9590162509f956ff79851fb1bc0ced6646f5d5ba7eae847a9f21c92c39437c',
owner: '32911fabcf90b9731a152d2a3a75fcbb0a46c78e2f502678bae44c3d6823b4ce',
ticker: '새로운경제',
enabled: false,
),
];
List<ZanoAsset> get initialZanoAssets => _defaultAssets.map(
(asset) {
String? iconPath;
try {
iconPath = CryptoCurrency.all.firstWhere((element) => element.title.toUpperCase() == asset.title.toUpperCase()).iconPath;
} catch (_) {}
return ZanoAsset.copyWith(asset, iconPath, 'ZANO');
},
).toList();
}

View file

@ -1,9 +1,5 @@
import 'dart:convert';
import 'package:cw_core/pending_transaction.dart'; import 'package:cw_core/pending_transaction.dart';
import 'package:cw_zano/api/exceptions/transfer_exception.dart';
import 'package:cw_zano/api/model/destination.dart'; import 'package:cw_zano/api/model/destination.dart';
import 'package:cw_zano/api/model/transfer_params.dart';
import 'package:cw_zano/api/model/transfer_result.dart'; import 'package:cw_zano/api/model/transfer_result.dart';
import 'package:cw_zano/zano_formatter.dart'; import 'package:cw_zano/zano_formatter.dart';
import 'package:cw_zano/zano_wallet.dart'; import 'package:cw_zano/zano_wallet.dart';
@ -45,33 +41,7 @@ class PendingZanoTransaction with PendingTransaction {
@override @override
Future<void> commit() async { Future<void> commit() async {
final params = TransferParams( await zanoWallet.transfer(destinations, fee, comment);
destinations: destinations, await zanoWallet.fetchTransactions();
fee: fee,
mixin: zanoMixinValue,
paymentId: '',
comment: comment,
pushPayer: false,
hideReceiver: true,
);
final result = await zanoWallet.invokeMethod('transfer', params);
final map = jsonDecode(result);
final resultMap = map['result'] as Map<String, dynamic>?;
if (resultMap != null) {
final transferResultMap = resultMap['result'] as Map<String, dynamic>?;
if (transferResultMap != null) {
transferResult = TransferResult.fromJson(transferResultMap);
print('transfer success hash ${transferResult!.txHash}');
await zanoWallet.fetchTransactions();
} else {
final errorCode = resultMap['error']['code'];
final code = errorCode is int ? errorCode.toString() : errorCode as String? ?? '';
final message = resultMap['error']['message'] as String? ?? '';
print('transfer error $code $message');
throw TransferException(code, message);
}
}
} }
} }

View file

@ -41,7 +41,7 @@ class ZanoAsset extends CryptoCurrency with HiveObjectMixin {
this.ticker = '', this.ticker = '',
required this.assetId, required this.assetId,
this.decimalPoint = ZanoFormatter.defaultDecimalPoint, this.decimalPoint = ZanoFormatter.defaultDecimalPoint,
bool enabled = true, bool enabled = false,
this.iconPath, this.iconPath,
this.tag, this.tag,
this.owner = defaultOwner, this.owner = defaultOwner,
@ -59,16 +59,6 @@ class ZanoAsset extends CryptoCurrency with HiveObjectMixin {
decimals: decimalPoint, decimals: decimalPoint,
); );
// ZanoAsset.copyWithCurrency(CryptoCurrency other, String? icon, String? tag, String? assetId, {bool enabled = false}):
// ZanoAsset(assetId: assetId, );
// // this.fullName = other.fullName ?? other.title,
// // this.ticker = other.title,
// // this.decimalPoint = other.decimals,
// // this.assetId = assetId,
// // this.iconPath = icon,
// // this.tag = tag,
// // this._enabled = enabled;
ZanoAsset.copyWith(ZanoAsset other, String? icon, String? tag, {String? assetId, bool enabled = false}) ZanoAsset.copyWith(ZanoAsset other, String? icon, String? tag, {String? assetId, bool enabled = false})
: this.fullName = other.fullName, : this.fullName = other.fullName,
this.ticker = other.ticker, this.ticker = other.ticker,

View file

@ -10,32 +10,30 @@ import 'package:cw_core/pending_transaction.dart';
import 'package:cw_core/sync_status.dart'; import 'package:cw_core/sync_status.dart';
import 'package:cw_core/transaction_priority.dart'; import 'package:cw_core/transaction_priority.dart';
import 'package:cw_core/wallet_base.dart'; 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_info.dart';
import 'package:cw_zano/api/api_calls.dart'; import 'package:cw_zano/api/model/create_wallet_result.dart';
import 'package:cw_zano/api/model/destination.dart'; import 'package:cw_zano/api/model/destination.dart';
import 'package:cw_zano/api/model/get_wallet_status_result.dart'; import 'package:cw_zano/api/model/get_wallet_status_result.dart';
import 'package:cw_zano/api/model/transfer.dart'; import 'package:cw_zano/api/model/transfer.dart';
import 'package:cw_zano/model/zano_wallet_keys.dart';
import 'package:cw_zano/model/zano_transaction_creation_exception.dart';
import 'package:cw_zano/model/pending_zano_transaction.dart'; import 'package:cw_zano/model/pending_zano_transaction.dart';
import 'package:cw_zano/model/zano_asset.dart'; import 'package:cw_zano/model/zano_asset.dart';
import 'package:cw_zano/model/zano_balance.dart'; import 'package:cw_zano/model/zano_balance.dart';
import 'package:cw_zano/zano_formatter.dart'; import 'package:cw_zano/model/zano_transaction_creation_exception.dart';
import 'package:cw_zano/model/zano_transaction_credentials.dart'; import 'package:cw_zano/model/zano_transaction_credentials.dart';
import 'package:cw_zano/zano_transaction_history.dart';
import 'package:cw_zano/model/zano_transaction_info.dart'; import 'package:cw_zano/model/zano_transaction_info.dart';
import 'package:cw_zano/model/zano_wallet_keys.dart';
import 'package:cw_zano/zano_formatter.dart';
import 'package:cw_zano/zano_transaction_history.dart';
import 'package:cw_zano/zano_wallet_addresses.dart'; import 'package:cw_zano/zano_wallet_addresses.dart';
import 'package:cw_zano/zano_wallet_api.dart'; import 'package:cw_zano/zano_wallet_api.dart';
import 'package:cw_zano/zano_wallet_service.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:hive/hive.dart'; import 'package:hive/hive.dart';
import 'package:mobx/mobx.dart'; import 'package:mobx/mobx.dart';
import 'default_zano_assets.dart';
part 'zano_wallet.g.dart'; part 'zano_wallet.g.dart';
const int zanoMixinValue = 10;
class ZanoWallet = ZanoWalletBase with _$ZanoWallet; class ZanoWallet = ZanoWalletBase with _$ZanoWallet;
abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHistory, ZanoTransactionInfo> with Store, ZanoWalletApi { abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHistory, ZanoTransactionInfo> with Store, ZanoWalletApi {
@ -62,6 +60,7 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
static const String zanoAssetId = 'd6329b5b1f7c0805b5c345f4957554002a2f557845f64d7645dae0e051a6498a'; static const String zanoAssetId = 'd6329b5b1f7c0805b5c345f4957554002a2f557845f64d7645dae0e051a6498a';
late final Box<ZanoAsset> zanoAssetsBox; late final Box<ZanoAsset> zanoAssetsBox;
List<ZanoAsset> get zanoAssets => zanoAssetsBox.values.toList(); List<ZanoAsset> get zanoAssets => zanoAssetsBox.values.toList();
// final Map<String, ZanoAsset> zanoAssets = {};
//zano_wallet.SyncListener? _listener; //zano_wallet.SyncListener? _listener;
// ReactionDisposer? _onAccountChangeReaction; // ReactionDisposer? _onAccountChangeReaction;
@ -103,6 +102,66 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
setPassword(password); setPassword(password);
} }
static Future<ZanoWallet> create({required WalletCredentials credentials}) async {
final wallet = ZanoWallet(credentials.walletInfo!);
await wallet.connectToNode(node: Node());
final path = await pathForWallet(name: credentials.name, type: credentials.walletInfo!.type);
final createWalletResult = await wallet.createWallet(path, credentials.password!);
_parseCreateWalletResult(createWalletResult, wallet);
//await wallet.store(); // TODO: unnecessary here?
await wallet.init(createWalletResult.wi.address);
return wallet;
}
static Future<ZanoWallet> restore({required ZanoRestoreWalletFromSeedCredentials credentials}) async {
final wallet = ZanoWallet(credentials.walletInfo!);
await wallet.connectToNode(node: Node());
final path = await pathForWallet(name: credentials.name, type: credentials.walletInfo!.type);
final createWalletResult = await wallet.restoreWalletFromSeed(path, credentials.password!, credentials.mnemonic);
_parseCreateWalletResult(createWalletResult, wallet);
//await wallet.store(); // TODO: unnecessary here?
await wallet.init(createWalletResult.wi.address);
return wallet;
}
static Future<ZanoWallet> open({required String name, required String password, required WalletInfo walletInfo}) async {
final path = await pathForWallet(name: name, type: walletInfo.type);
final wallet = ZanoWallet(walletInfo);
await wallet.connectToNode(node: Node());
final createWalletResult = await wallet.loadWallet(path, password);
_parseCreateWalletResult(createWalletResult, wallet);
//await wallet.store(); // TODO: unnecessary here?
await wallet.init(createWalletResult.wi.address);
return wallet;
}
static void _parseCreateWalletResult(CreateWalletResult result, ZanoWallet wallet) {
wallet.hWallet = result.walletId;
wallet.walletAddresses.address = result.wi.address;
for (final item in result.wi.balances) {
if (item.assetInfo.ticker == 'ZANO') {
wallet.balance[CryptoCurrency.zano] = ZanoBalance(
total: item.total,
unlocked: item.unlocked,
decimalPoint: ZanoFormatter.defaultDecimalPoint,
);
} 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,
decimalPoint: asset.decimalPoint,
);
}
}
}
}
if (result.recentHistory.history != null) {
wallet.transfers = result.recentHistory.history!;
}
}
@override @override
void close() { void close() {
closeWallet(); closeWallet();
@ -221,6 +280,20 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
} }
}, },
); );
/*final Map<String, ZanoTransactionInfo> result = {};
for (final item in transfers) {
if (item.subtransfers.first.assetId == zanoAssetId) {
result[item.txHash] = ZanoTransactionInfo.fromTransfer(item, 'ZANO', ZanoFormatter.defaultDecimalPoint);
} else {
if (!zanoAssets.containsKey(item.subtransfers.first.assetId)) {
print('no such asset ${item.subtransfers.first.assetId}');
} else {
final asset = zanoAssets[item.subtransfers.first.assetId]!;
result[item.txHash] = ZanoTransactionInfo.fromTransfer(item, asset.ticker, asset.decimalPoint);
}
}
}
return result;*/
} catch (e) { } catch (e) {
print(e); print(e);
return {}; return {};
@ -237,21 +310,11 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
} }
await walletAddresses.init(); await walletAddresses.init();
await walletAddresses.updateAddress(address); await walletAddresses.updateAddress(address);
///balance.addAll(getZanoBalance(/**accountIndex: walletAddresses.account?.id ?? 0*/));
//_setListeners(); //_setListeners();
await updateTransactions(); await updateTransactions();
_autoSaveTimer = Timer.periodic(Duration(seconds: _autoSaveInterval), (_) async => await save()); _autoSaveTimer = Timer.periodic(Duration(seconds: _autoSaveInterval), (_) async => await save());
} }
String loadWallet(String path, String password) {
print('load_wallet path $path password $password');
final result = ApiCalls.loadWallet(path: path, password: password);
print('load_wallet result $result');
return result;
}
@override @override
Future<void> renameWalletFiles(String newWalletName) async { Future<void> renameWalletFiles(String newWalletName) async {
final currentWalletPath = await pathForWallet(name: name, type: type); final currentWalletPath = await pathForWallet(name: name, type: type);
@ -277,6 +340,8 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
} }
@override @override
Future<void> rescan({required int height}) => throw UnimplementedError();
/*@override
Future<void> rescan({required int height}) async { Future<void> rescan({required int height}) async {
walletInfo.restoreHeight = height; walletInfo.restoreHeight = height;
walletInfo.isRecovery = true; walletInfo.isRecovery = true;
@ -287,25 +352,18 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
await _askForUpdateTransactionHistory(); await _askForUpdateTransactionHistory();
await save(); await save();
await walletInfo.save(); await walletInfo.save();
} }*/
@override @override
Future<void> save() async { Future<void> save() async {
try { try {
await walletAddresses.updateAddressesInBox();
await backupWalletFiles(name);
await store(); await store();
await walletAddresses.updateAddressesInBox();
} catch (e) { } catch (e) {
print('Error while saving Zano wallet file ${e.toString()}'); print('Error while saving Zano wallet file ${e.toString()}');
} }
} }
Future<void> setAsRecovered() async {
walletInfo.isRecovery = false;
await walletInfo.save();
}
bool _calledOnce = false;
int _counter = 0; int _counter = 0;
@override @override
@ -320,11 +378,12 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
onNewTransaction?.call(); onNewTransaction?.call();
}*/ }*/
final walletStatus = getWalletStatus(); final walletStatus = await getWalletStatus();
_updateSyncProgress(walletStatus); _updateSyncProgress(walletStatus);
// You can call getWalletInfo ONLY if getWalletStatus returns NOT is in long refresh and wallet state is 2 (ready)
// we can call getWalletInfo ONLY if getWalletStatus returns NOT is in long refresh and wallet state is 2 (ready)
if (!walletStatus.isInLongRefresh && walletStatus.walletState == 2) { if (!walletStatus.isInLongRefresh && walletStatus.walletState == 2) {
final walletInfo = getWalletInfo(); final walletInfo = await getWalletInfo();
seed = walletInfo.wiExtended.seed; seed = walletInfo.wiExtended.seed;
keys = ZanoWalletKeys( keys = ZanoWalletKeys(
privateSpendKey: walletInfo.wiExtended.spendPrivateKey, privateSpendKey: walletInfo.wiExtended.spendPrivateKey,
@ -333,32 +392,43 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
publicViewKey: walletInfo.wiExtended.viewPublicKey, publicViewKey: walletInfo.wiExtended.viewPublicKey,
); );
final whitelists = await getAssetsWhitelist();
// TODO: here should be synchronization of whitelists
// for (final item in whitelists) {
// if (!zanoAssets.containsKey(item.assetId)) zanoAssets[item.assetId] = item;
// }
// // removing assets missing in whitelists (in case some were removed since last time)
// zanoAssets.removeWhere((key, _) => !whitelists.any((element) => element.assetId == key));
// matching balances and whitelists
// 1. show only balances available in whitelists
// 2. set whitelists available in balances as 'enabled' ('disabled' by default)
for (final item in walletInfo.wi.balances) { for (final item in walletInfo.wi.balances) {
if (item.assetInfo.ticker == 'ZANO') { if (item.assetInfo.ticker == 'ZANO') {
balance[CryptoCurrency.zano] = ZanoBalance(total: item.total, unlocked: item.unlocked, decimalPoint: ZanoFormatter.defaultDecimalPoint); balance[CryptoCurrency.zano] = ZanoBalance(total: item.total, unlocked: item.unlocked, decimalPoint: ZanoFormatter.defaultDecimalPoint);
} else { } else {
for (final asset in balance.keys) { final asset = zanoAssetsBox.get(item.assetInfo.assetId);
if (asset is ZanoAsset && asset.assetId == item.assetInfo.assetId) { if (asset == null) {
balance[asset] = ZanoBalance(total: item.total, unlocked: item.unlocked, decimalPoint: asset.decimalPoint); debugPrint('balance for an unknown asset ${item.assetInfo.assetId}');
} continue;
} }
if (balance.keys.any((element) => element is ZanoAsset && element.assetId == item.assetInfo.assetId)) {
balance[balance.keys.firstWhere((element) => element is ZanoAsset && element.assetId == item.assetInfo.assetId)] =
ZanoBalance(total: item.total, unlocked: item.unlocked, decimalPoint: asset.decimalPoint);
} else {
balance[asset] = ZanoBalance(total: item.total, unlocked: item.unlocked, decimalPoint: asset.decimalPoint);
}
//balance[asset] = ZanoBalance(total: item.total, unlocked: item.unlocked, decimalPoint: asset.decimalPoint);
asset.enabled = true;
} }
} }
// removing balances for assets missing in wallet info balances (in case they were removed for some reason)
balance.removeWhere(
(key, _) =>
key != CryptoCurrency.zano && !walletInfo.wi.balances.any((element) => element.assetInfo.assetId == (key as ZanoAsset).assetId),
);
//await getAssetsWhitelist(); //if (_counter++ % 10 == 0) await _askForUpdateTransactionHistory();
if (!_calledOnce) {
//await addAssetsWhitelist('00112233445566778899aabbccddeeff00112233445566778899aabbccddeeff');
//await removeAssetsWhitelist('cc4e69455e63f4a581257382191de6856c2156630b3fba0db4bdd73ffcfb36b6');
//await removeAssetsWhitelist('bb9590162509f956ff79851fb1bc0ced6646f5d5ba7eae847a9f21c92c39437c');
//await removeAssetsWhitelist('');
_calledOnce = true;
} else {
await getAssetsWhitelist();
}
// if (++_counter >= 10) {
// await getAssetsWhitelist();
// _counter = 0;
// }
} }
}); });
} catch (e) { } catch (e) {
@ -387,14 +457,6 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
} }
} }
void addInitialAssets() {
final initialZanoAssets = DefaultZanoAssets().initialZanoAssets;
for (var token in initialZanoAssets) {
zanoAssetsBox.put(token.assetId, token);
}
}
Future<CryptoCurrency> addZanoAssetById(String assetId) async { Future<CryptoCurrency> addZanoAssetById(String assetId) async {
if (zanoAssetsBox.containsKey(assetId)) { if (zanoAssetsBox.containsKey(assetId)) {
throw 'zano asset with id $assetId already added'; throw 'zano asset with id $assetId already added';
@ -407,8 +469,9 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
try { try {
iconPath = CryptoCurrency.all.firstWhere((element) => element.title.toUpperCase() == assetDescriptor.title.toUpperCase()).iconPath; iconPath = CryptoCurrency.all.firstWhere((element) => element.title.toUpperCase() == assetDescriptor.title.toUpperCase()).iconPath;
} catch (_) {} } catch (_) {}
// TODO: copywith two times. was it intended
final asset = ZanoAsset.copyWith(assetDescriptor, iconPath, 'ZANO', assetId: assetId, enabled: true); final asset = ZanoAsset.copyWith(assetDescriptor, iconPath, 'ZANO', assetId: assetId, enabled: true);
await zanoAssetsBox.put(asset.assetId, ZanoAsset.copyWith(asset, iconPath, 'ZANO')); zanoAssetsBox.put(asset.assetId, ZanoAsset.copyWith(asset, iconPath, 'ZANO'));
balance[asset] = ZanoBalance(total: 0, unlocked: 0, decimalPoint: asset.decimalPoint); balance[asset] = ZanoBalance(total: 0, unlocked: 0, decimalPoint: asset.decimalPoint);
return asset; return asset;
} }
@ -418,7 +481,7 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
try { try {
iconPath = CryptoCurrency.all.firstWhere((element) => element.title.toUpperCase() == asset.title.toUpperCase()).iconPath; iconPath = CryptoCurrency.all.firstWhere((element) => element.title.toUpperCase() == asset.title.toUpperCase()).iconPath;
} catch (_) {} } catch (_) {}
await zanoAssetsBox.put(asset.assetId, ZanoAsset.copyWith(asset, iconPath, 'ZANO')); zanoAssetsBox.put(asset.assetId, ZanoAsset.copyWith(asset, iconPath, 'ZANO'));
if (asset.enabled) { if (asset.enabled) {
final assetDescriptor = await addAssetsWhitelist(asset.assetId); final assetDescriptor = await addAssetsWhitelist(asset.assetId);
if (assetDescriptor == null) { if (assetDescriptor == null) {
@ -460,12 +523,8 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
void _onNewBlock(int height, int blocksLeft, double ptc) async { void _onNewBlock(int height, int blocksLeft, double ptc) async {
try { try {
if (walletInfo.isRecovery) {
await _askForUpdateTransactionHistory();
/*walletAddresses.accountList.update();*/
}
if (blocksLeft < 1000) { if (blocksLeft < 1000) {
// TODO: we can't update transactions history before loading all balances and whitelists
await _askForUpdateTransactionHistory(); await _askForUpdateTransactionHistory();
/*walletAddresses.accountList.update();*/ /*walletAddresses.accountList.update();*/
syncStatus = SyncedSyncStatus(); syncStatus = SyncedSyncStatus();
@ -474,10 +533,6 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
_hasSyncAfterStartup = true; _hasSyncAfterStartup = true;
await save(); await save();
} }
if (walletInfo.isRecovery) {
await setAsRecovered();
}
} else { } else {
syncStatus = SyncingSyncStatus(blocksLeft, ptc); syncStatus = SyncingSyncStatus(blocksLeft, ptc);
} }

View file

@ -6,7 +6,6 @@ part 'zano_wallet_addresses.g.dart';
class ZanoWalletAddresses = ZanoWalletAddressesBase with _$ZanoWalletAddresses; class ZanoWalletAddresses = ZanoWalletAddressesBase with _$ZanoWalletAddresses;
/**abstract class ZanoWalletAddressesBase extends WalletAddressesWithAccount<Account> with Store {*/
abstract class ZanoWalletAddressesBase extends WalletAddresses with Store { abstract class ZanoWalletAddressesBase extends WalletAddresses with Store {
ZanoWalletAddressesBase(WalletInfo walletInfo) ZanoWalletAddressesBase(WalletInfo walletInfo)
: address = '', : address = '',
@ -16,24 +15,10 @@ abstract class ZanoWalletAddressesBase extends WalletAddresses with Store {
@observable @observable
String address; String address;
// @override
/**@observable
Account? account;*/
/**@observable
Subaddress? subaddress;*/
/**ZanoSubaddressList subaddressList;*/
/**ZanoAccountList accountList;*/
@override @override
Future<void> init() async { Future<void> init() async {
/*accountList.update(); address = walletInfo.address;
account = accountList.accounts.first;*/ await updateAddressesInBox();
/**updateSubaddressList(accountIndex: account?.id ?? 0);*/
//address = walletInfo.address;
//await updateAddressesInBox();
} }
Future<void> updateAddress(String address) async { Future<void> updateAddress(String address) async {
@ -44,46 +29,11 @@ abstract class ZanoWalletAddressesBase extends WalletAddresses with Store {
@override @override
Future<void> updateAddressesInBox() async { Future<void> updateAddressesInBox() async {
try { try {
/**final _subaddressList = ZanoSubaddressList();*/
addressesMap.clear(); addressesMap.clear();
addressesMap[address] = ''; addressesMap[address] = '';
await saveAddressesInBox(); await saveAddressesInBox();
/*accountList.accounts.forEach((account) {
_subaddressList.update(accountIndex: account.id);
_subaddressList.subaddresses.forEach((subaddress) {
addressesMap[subaddress.address] = subaddress.label;
});
});
await saveAddressesInBox();*/
} catch (e) { } catch (e) {
print(e.toString()); print(e.toString());
} }
} }
// bool validate() {
// accountList.update();
// final accountListLength = accountList.accounts.length ?? 0;
// if (accountListLength <= 0) {
// return false;
// }
// /**subaddressList.update(accountIndex: accountList.accounts.first.id);
// final subaddressListLength = subaddressList.subaddresses.length ?? 0;
// if (subaddressListLength <= 0) {
// return false;
// }*/
// return true;
// }
/*void updateSubaddressList({required int accountIndex}) {
subaddressList.update(accountIndex: accountIndex);
subaddress = subaddressList.subaddresses.first;
address = subaddress!.address;
}*/
} }

View file

@ -1,8 +1,12 @@
import 'dart:convert'; import 'dart:convert';
import 'dart:io';
import 'package:cw_core/transaction_priority.dart'; import 'package:cw_core/transaction_priority.dart';
import 'package:cw_zano/api/api_calls.dart'; import 'package:cw_zano/api/api_calls.dart';
import 'package:cw_zano/api/consts.dart';
import 'package:cw_zano/api/model/asset_id_params.dart'; import 'package:cw_zano/api/model/asset_id_params.dart';
import 'package:cw_zano/api/model/create_wallet_result.dart';
import 'package:cw_zano/api/model/destination.dart';
import 'package:cw_zano/api/model/get_address_info_result.dart'; import 'package:cw_zano/api/model/get_address_info_result.dart';
import 'package:cw_zano/api/model/get_recent_txs_and_info_params.dart'; import 'package:cw_zano/api/model/get_recent_txs_and_info_params.dart';
import 'package:cw_zano/api/model/get_wallet_info_result.dart'; import 'package:cw_zano/api/model/get_wallet_info_result.dart';
@ -10,18 +14,25 @@ import 'package:cw_zano/api/model/get_wallet_status_result.dart';
import 'package:cw_zano/api/model/proxy_to_daemon_params.dart'; import 'package:cw_zano/api/model/proxy_to_daemon_params.dart';
import 'package:cw_zano/api/model/proxy_to_daemon_result.dart'; import 'package:cw_zano/api/model/proxy_to_daemon_result.dart';
import 'package:cw_zano/api/model/transfer.dart'; import 'package:cw_zano/api/model/transfer.dart';
import 'package:cw_zano/api/model/transfer_params.dart';
import 'package:cw_zano/api/model/transfer_result.dart';
import 'package:cw_zano/model/zano_asset.dart'; import 'package:cw_zano/model/zano_asset.dart';
import 'package:cw_zano/zano_wallet_exceptions.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:path_provider/path_provider.dart';
import 'api/model/store_result.dart'; import 'api/model/store_result.dart';
enum _LogType { none, simple, json } //enum _LogType { none, simple, json }
mixin ZanoWalletApi { mixin ZanoWalletApi {
static const _defaultNodeUri = '195.201.107.230:33336'; static const _defaultNodeUri = '195.201.107.230:33336';
static const _statusDelivered = 'delivered'; static const _statusDelivered = 'delivered';
static const _maxAttempts = 10; static const _maxAttempts = 10;
static const _logType = _LogType.simple; //static const _logType = _LogType.json;
static const _logInfo = true;
static const _logJson = false;
static const int _zanoMixinValue = 10;
int _hWallet = 0; int _hWallet = 0;
@ -37,43 +48,46 @@ mixin ZanoWalletApi {
void closeWallet() => ApiCalls.closeWallet(hWallet: hWallet); void closeWallet() => ApiCalls.closeWallet(hWallet: hWallet);
Future<bool> setupNode() async => ApiCalls.setupNode( Future<bool> setupNode() async {
address: _defaultNodeUri, debugPrint('[info] init $_defaultNodeUri');
login: '', final result = ApiCalls.setupNode(
password: '', address: _defaultNodeUri,
useSSL: false, login: '',
isLightWallet: false, password: '',
); useSSL: false,
isLightWallet: false,
GetWalletInfoResult getWalletInfo() { );
final json = ApiCalls.getWalletInfo(hWallet); debugPrint('[info] init result $result');
final result = GetWalletInfoResult.fromJson(jsonDecode(json) as Map<String, dynamic>);
switch (_logType) {
case _LogType.json:
debugPrint('get_wallet_info $json');
break;
case _LogType.simple:
debugPrint('get_wallet_info got ${result.wi.balances.length} balances: ${result.wi.balances}');
}
return result; return result;
} }
GetWalletStatusResult getWalletStatus() { Future<GetWalletInfoResult> getWalletInfo() async {
final json = ApiCalls.getWalletInfo(hWallet);
final result = GetWalletInfoResult.fromJson(jsonDecode(json) as Map<String, dynamic>);
if (_logJson) debugPrint('get_wallet_info $json');
await _writeLog('get_wallet_info', 'get_wallet_info result $json');
if (_logInfo)
debugPrint('[info] get_wallet_info got ${result.wi.balances.length} balances: ${result.wi.balances} seed: ${_shorten(result.wiExtended.seed)}');
return result;
}
Future<GetWalletStatusResult> getWalletStatus() async {
final json = ApiCalls.getWalletStatus(hWallet: hWallet); final json = ApiCalls.getWalletStatus(hWallet: hWallet);
final status = GetWalletStatusResult.fromJson(jsonDecode(json) as Map<String, dynamic>); if (json == Consts.errorWalletWrongId) {
switch (_logType) { print('wrong wallet id');
case _LogType.json: throw ZanoWalletException('Wrong wallet id');
debugPrint('get_wallet_status $json');
break;
case _LogType.simple:
debugPrint(
'get_wallet_status connected: ${status.isDaemonConnected} in refresh: ${status.isInLongRefresh} wallet state: ${status.walletState}');
} }
final status = GetWalletStatusResult.fromJson(jsonDecode(json) as Map<String, dynamic>);
if (_logJson) debugPrint('get_wallet_status $json');
await _writeLog('get_wallet_status', 'get_wallet_status result $json');
if (_logInfo)
debugPrint(
'[info] get_wallet_status connected: ${status.isDaemonConnected} in refresh: ${status.isInLongRefresh} progress: ${status.progress} wallet state: ${status.walletState}');
return status; return status;
} }
Future<String> invokeMethod(String methodName, Object params) async { Future<String> invokeMethod(String methodName, Object params) async {
await _writeLog(methodName, 'invoke method $methodName params: ${jsonEncode(params)} hWallet: $hWallet');
var invokeResult = var invokeResult =
ApiCalls.asyncCall(methodName: 'invoke', hWallet: hWallet, params: '{"method": "$methodName","params": ${jsonEncode(params)}}'); ApiCalls.asyncCall(methodName: 'invoke', hWallet: hWallet, params: '{"method": "$methodName","params": ${jsonEncode(params)}}');
var map = jsonDecode(invokeResult) as Map<String, dynamic>; var map = jsonDecode(invokeResult) as Map<String, dynamic>;
@ -85,17 +99,19 @@ mixin ZanoWalletApi {
final result = ApiCalls.tryPullResult(jobId); final result = ApiCalls.tryPullResult(jobId);
map = jsonDecode(result) as Map<String, dynamic>; map = jsonDecode(result) as Map<String, dynamic>;
if (map['status'] != null && map['status'] == _statusDelivered && map['result'] != null) { if (map['status'] != null && map['status'] == _statusDelivered && map['result'] != null) {
await _writeLog(methodName, 'invoke method $methodName result $result');
return result; return result;
} }
} while (++attempts < _maxAttempts); } while (++attempts < _maxAttempts);
} }
await _writeLog(methodName, 'invoke method $methodName result: $invokeResult');
return invokeResult; return invokeResult;
} }
Future<List<ZanoAsset>> getAssetsWhitelist() async { Future<List<ZanoAsset>> getAssetsWhitelist() async {
try { try {
final json = await invokeMethod('assets_whitelist_get', '{}'); final json = await invokeMethod('assets_whitelist_get', '{}');
/*if (_logType == _LogType.json)*/ debugPrint('assets_whitelist_get $json'); if (_logJson) debugPrint('assets_whitelist_get $json');
final map = jsonDecode(json) as Map<String, dynamic>?; final map = jsonDecode(json) as Map<String, dynamic>?;
_checkForErrors(map); _checkForErrors(map);
List<ZanoAsset> assets(String type) => List<ZanoAsset> assets(String type) =>
@ -103,13 +119,13 @@ mixin ZanoWalletApi {
final localWhitelist = assets('local_whitelist'); final localWhitelist = assets('local_whitelist');
final globalWhitelist = assets('global_whitelist'); final globalWhitelist = assets('global_whitelist');
final ownAssets = assets('own_assets'); final ownAssets = assets('own_assets');
if (_logType == _LogType.simple) if (_logInfo)
print('assets_whitelist_get got local whitelist: ${localWhitelist.length} ($localWhitelist); ' print('[info] assets_whitelist_get got local whitelist: ${localWhitelist.length} ($localWhitelist); '
'global whitelist: ${globalWhitelist.length} ($globalWhitelist); ' 'global whitelist: ${globalWhitelist.length} ($globalWhitelist); '
'own assets: ${ownAssets.length} ($ownAssets)'); 'own assets: ${ownAssets.length} ($ownAssets)');
return [...localWhitelist, ...globalWhitelist, ...ownAssets]; return [...localWhitelist, ...globalWhitelist, ...ownAssets];
} catch (e) { } catch (e) {
print(e.toString()); print('[error] assets_whitelist_get $e');
return []; return [];
} }
} }
@ -117,19 +133,19 @@ mixin ZanoWalletApi {
Future<ZanoAsset?> addAssetsWhitelist(String assetId) async { Future<ZanoAsset?> addAssetsWhitelist(String assetId) async {
try { try {
final json = await invokeMethod('assets_whitelist_add', AssetIdParams(assetId: assetId)); final json = await invokeMethod('assets_whitelist_add', AssetIdParams(assetId: assetId));
if (_logType == _LogType.json) print('assets_whitelist_add $assetId $json'); if (_logJson) print('assets_whitelist_add $assetId $json');
final map = jsonDecode(json) as Map<String, dynamic>?; final map = jsonDecode(json) as Map<String, dynamic>?;
_checkForErrors(map); _checkForErrors(map);
if (map!['result']!['result']!['status']! == 'OK') { if (map!['result']!['result']!['status']! == 'OK') {
final assetDescriptor = ZanoAsset.fromJson(map['result']!['result']!['asset_descriptor']! as Map<String, dynamic>); final assetDescriptor = ZanoAsset.fromJson(map['result']!['result']!['asset_descriptor']! as Map<String, dynamic>);
if (_logType == _LogType.simple) print('assets_whitelist_add added ${assetDescriptor.fullName} ${assetDescriptor.ticker}'); if (_logInfo) print('[info] assets_whitelist_add added ${assetDescriptor.fullName} ${assetDescriptor.ticker}');
return assetDescriptor; return assetDescriptor;
} else { } else {
if (_logType == _LogType.simple) print('assets_whitelist_add status ${map['result']!['result']!['status']!}'); if (_logInfo) print('[info] assets_whitelist_add status ${map['result']!['result']!['status']!}');
return null; return null;
} }
} catch (e) { } catch (e) {
print(e.toString()); print('[error] assets_whitelist_add $e');
return null; return null;
} }
} }
@ -137,13 +153,13 @@ mixin ZanoWalletApi {
Future<bool> removeAssetsWhitelist(String assetId) async { Future<bool> removeAssetsWhitelist(String assetId) async {
try { try {
final json = await invokeMethod('assets_whitelist_remove', AssetIdParams(assetId: assetId)); final json = await invokeMethod('assets_whitelist_remove', AssetIdParams(assetId: assetId));
if (_logType == _LogType.json) print('assets_whitelist_remove $assetId $json'); if (_logJson) print('assets_whitelist_remove $assetId $json');
final map = jsonDecode(json) as Map<String, dynamic>?; final map = jsonDecode(json) as Map<String, dynamic>?;
_checkForErrors(map); _checkForErrors(map);
if (_logType == _LogType.simple) print('assets_whitelist_remove status ${map!['result']!['result']!['status']!}'); if (_logInfo) print('[info] assets_whitelist_remove status ${map!['result']!['result']!['status']!}');
return (map!['result']!['result']!['status']! == 'OK'); return (map!['result']!['result']!['status']! == 'OK');
} catch (e) { } catch (e) {
print(e.toString()); print('[error] assets_whitelist_remove $e');
return false; return false;
} }
} }
@ -159,21 +175,21 @@ mixin ZanoWalletApi {
final methodName = 'get_asset_info'; final methodName = 'get_asset_info';
final params = AssetIdParams(assetId: assetId); final params = AssetIdParams(assetId: assetId);
final result = await _proxyToDaemon('/json_rpc', '{"method": "$methodName","params": ${jsonEncode(params)}}'); final result = await _proxyToDaemon('/json_rpc', '{"method": "$methodName","params": ${jsonEncode(params)}}');
if (_logType == _LogType.json) print('$methodName $assetId ${result?.body}'); if (_logJson) print('$methodName $assetId ${result?.body}');
if (result == null) { if (result == null) {
debugPrint('get_asset_info empty result'); debugPrint('get_asset_info empty result');
return null; return null;
} }
final map = jsonDecode(result.body) as Map<String, dynamic>?; final map = jsonDecode(result.body) as Map<String, dynamic>?;
if (map!['error'] != null) { if (map!['error'] != null) {
if (_logType == _LogType.simple) print('get_asset_info $assetId error ${map['error']!['code']} ${map['error']!['message']}'); if (_logInfo) print('[info] get_asset_info $assetId error ${map['error']!['code']} ${map['error']!['message']}');
return null; return null;
} else if (map['result']!['status']! == 'OK') { } else if (map['result']!['status']! == 'OK') {
final assetDescriptor = ZanoAsset.fromJson(map['result']!['asset_descriptor']! as Map<String, dynamic>); final assetDescriptor = ZanoAsset.fromJson(map['result']!['asset_descriptor']! as Map<String, dynamic>);
if (_logType == _LogType.simple) print('get_asset_info $assetId ${assetDescriptor.fullName} ${assetDescriptor.ticker}'); if (_logInfo) print('[info] get_asset_info $assetId ${assetDescriptor.fullName} ${assetDescriptor.ticker}');
return assetDescriptor; return assetDescriptor;
} else { } else {
if (_logType == _LogType.simple) print('get_asset_info $assetId status ${map['result']!['status']!}'); if (_logInfo) print('[info] get_asset_info $assetId status ${map['result']!['status']!}');
return null; return null;
} }
} }
@ -185,7 +201,7 @@ mixin ZanoWalletApi {
_checkForErrors(map); _checkForErrors(map);
return StoreResult.fromJson(map!['result']['result'] as Map<String, dynamic>); return StoreResult.fromJson(map!['result']['result'] as Map<String, dynamic>);
} catch (e) { } catch (e) {
print(e); print('[error] store $e');
return null; return null;
} }
} }
@ -193,17 +209,18 @@ mixin ZanoWalletApi {
Future<List<Transfer>> getRecentTxsAndInfo() async { Future<List<Transfer>> getRecentTxsAndInfo() async {
try { try {
final json = await invokeMethod('get_recent_txs_and_info', GetRecentTxsAndInfoParams(offset: 0, count: 30)); final json = await invokeMethod('get_recent_txs_and_info', GetRecentTxsAndInfoParams(offset: 0, count: 30));
//debugPrint('get_recent_txs_and_info $json'); if (_logJson) debugPrint('get_recent_txs_and_info $json');
final map = jsonDecode(json) as Map<String, dynamic>?; final map = jsonDecode(json) as Map<String, dynamic>?;
_checkForErrors(map); _checkForErrors(map);
final transfers = map?['result']?['result']?['transfers'] as List<dynamic>?; final transfers = map?['result']?['result']?['transfers'] as List<dynamic>?;
if (transfers == null) { if (transfers == null) {
print('get_recent_txs_and_info empty transfers'); if (_logInfo) print('[info] get_recent_txs_and_info empty transfers');
return []; return [];
} }
if (_logInfo) print('[info] get_recent_txs_and_info transfers: ${transfers.length}');
return transfers.map((e) => Transfer.fromJson(e as Map<String, dynamic>)).toList(); return transfers.map((e) => Transfer.fromJson(e as Map<String, dynamic>)).toList();
} catch (e) { } catch (e) {
print(e); print('[error] get_recent_txs_and_info $e');
return []; return [];
} }
} }
@ -212,20 +229,140 @@ mixin ZanoWalletApi {
jsonDecode(ApiCalls.getAddressInfo(address: address)) as Map<String, dynamic>, jsonDecode(ApiCalls.getAddressInfo(address: address)) as Map<String, dynamic>,
); );
String _shorten(String s) => s.length > 10 ? '${s.substring(0, 4)}...${s.substring(s.length - 4)}' : s;
Future<CreateWalletResult> createWallet(String path, String password) async {
if (_logInfo) debugPrint('[info] create_wallet path $path password ${_shorten(password)}');
await _writeLog('create_wallet', 'create_wallet path $path password ${_shorten(password)}');
final json = ApiCalls.createWallet(path: path, password: password);
if (_logJson) debugPrint('create_wallet $json');
await _writeLog('create_wallet', 'create_wallet result $json');
final map = jsonDecode(json) as Map<String, dynamic>?;
if (map?['error'] != null) {
final code = map!['error']!['code'] ?? '';
final message = map['error']!['message'] ?? '';
throw ZanoWalletException('Error creating wallet file, $message ($code)');
}
if (map?['result'] == null) {
throw ZanoWalletException('Error creating wallet file, empty response');
}
final result = CreateWalletResult.fromJson(map!['result'] as Map<String, dynamic>);
if (_logInfo) debugPrint('[info] create_wallet ${result.name} ${result.seed}');
return result;
}
Future<CreateWalletResult> restoreWalletFromSeed(String path, String password, String seed) async {
if (_logInfo) debugPrint('[info] restore_wallet path $path password ${_shorten(password)} seed ${_shorten(seed)}');
await _writeLog('restore_wallet', 'restore_wallet path $path password ${_shorten(password)} seed ${_shorten(seed)}');
final json = ApiCalls.restoreWalletFromSeed(path: path, password: password, seed: seed);
if (_logJson) debugPrint('restore_wallet $json');
await _writeLog('restore_wallet', 'restore_wallet result $json');
final map = jsonDecode(json) as Map<String, dynamic>?;
if (map?['error'] != null) {
final code = map!['error']!['code'] ?? '';
final message = map['error']!['message'] ?? '';
if (code == Consts.errorWrongSeed) {
throw RestoreFromKeysException('Error restoring wallet, wrong seed');
} else if (code == Consts.errorAlreadyExists) {
throw RestoreFromKeysException('Error restoring wallet, already exists');
}
throw RestoreFromKeysException('Error restoring wallet, $message ($code)');
}
if (map?['result'] == null) {
throw RestoreFromKeysException('Error restoring wallet, empty response');
}
final result = CreateWalletResult.fromJson(map!['result'] as Map<String, dynamic>);
if (_logInfo) debugPrint('[info] restore_wallet ${result.name} ${result.wi.address}');
return result;
}
Future<CreateWalletResult> loadWallet(String path, String password, [bool secondAttempt = false]) async {
if (_logInfo) debugPrint('[info] load_wallet path $path password ${_shorten(password)}');
await _writeLog('load_wallet', 'load_wallet path $path password ${_shorten(password)}');
final json = ApiCalls.loadWallet(path: path, password: password);
if (_logJson) debugPrint('load_wallet $json');
await _writeLog('load_wallet', 'load_wallet result $json');
final map = jsonDecode(json) as Map<String, dynamic>?;
if (map?['error'] != null) {
final code = map?['error']!['code'] ?? '';
final message = map?['error']!['message'] ?? '';
if (code == Consts.errorAlreadyExists && !secondAttempt) {
// TODO: that's not the best solution!
// already connected to this wallet. closing and attempting to reopen
debugPrint('already connected. closing and reopen wallet');
closeWallet();
await Future.delayed(const Duration(milliseconds: 500));
return await loadWallet(path, password, true);
}
throw ZanoWalletException('Error loading wallet, $message ($code)');
}
if (map?['result'] == null) {
throw ZanoWalletException('Error loading wallet, empty response');
}
final result = CreateWalletResult.fromJson(map!['result'] as Map<String, dynamic>);
if (_logInfo) debugPrint('[info] load_wallet ${result.name} ${result.wi.address}');
return result;
}
Future<TransferResult> transfer(List<Destination> destinations, BigInt fee, String comment) async {
final params = TransferParams(
destinations: destinations,
fee: fee,
mixin: _zanoMixinValue,
paymentId: '',
comment: comment,
pushPayer: false,
hideReceiver: true,
);
final json = await invokeMethod('transfer', params);
if (_logJson) debugPrint('transfer $json');
final map = jsonDecode(json);
final resultMap = map['result'] as Map<String, dynamic>?;
if (resultMap != null) {
final transferResultMap = resultMap['result'] as Map<String, dynamic>?;
if (transferResultMap != null) {
final transferResult = TransferResult.fromJson(transferResultMap);
debugPrint('transfer success hash ${transferResult.txHash}');
return transferResult;
} else {
final errorCode = resultMap['error']['code'];
final code = errorCode is int ? errorCode.toString() : errorCode as String? ?? '';
final message = resultMap['error']['message'] as String? ?? '';
debugPrint('transfer error $code $message');
throw TransferException('Transfer error, $message ($code)');
}
}
debugPrint('transfer error empty result');
throw TransferException('Transfer error, empty result');
}
void _checkForErrors(Map<String, dynamic>? map) { void _checkForErrors(Map<String, dynamic>? map) {
if (map == null) { if (map == null) {
throw 'empty response'; throw ZanoWalletException('Empty response');
} }
final result = map['result']; final result = map['result'];
if (result == null) { if (result == null) {
throw 'empty response'; throw ZanoWalletException('Empty response');
} }
if (result['error'] != null) { if (result['error'] != null) {
final code = result['error']!['code'] ?? ''; final code = result['error']!['code'] ?? '';
final message = result['error']!['message'] ?? ''; final message = result['error']!['message'] ?? '';
throw 'error $code $message'; throw ZanoWalletException('Error, $message ($code)');
}
}
Future<void> _writeLog(String method, String logMessage) async {
final dir = await getDownloadsDirectory();
final logFile = File('${dir!.path}/$method.txt');
final date = DateTime.now();
String twoDigits(int value) => value.toString().padLeft(2, '0');
String removeCRandLF(String input) => input.replaceAll(RegExp('\r|\n'), '');
await logFile.writeAsString('${twoDigits(date.hour)}:${twoDigits(date.minute)}:${twoDigits(date.second)} ${removeCRandLF(logMessage)}\n',
mode: FileMode.append);
RegExp regExp = RegExp(r'"fee":\s*(\d+(?:\.\d+)?)');
final matches = regExp.allMatches(logMessage);
if (matches.isNotEmpty) {
await logFile.writeAsString(' ' + matches.map((element) => '${element.group(0)}').join(', ') + '\n', mode: FileMode.append);
} }
} }
} }

View file

@ -0,0 +1,15 @@
class ZanoWalletException implements Exception {
final String message;
ZanoWalletException(this.message);
@override
String toString() => '${this.runtimeType} (message: $message)';
}
class RestoreFromKeysException extends ZanoWalletException {
RestoreFromKeysException(String message) : super(message);
}
class TransferException extends ZanoWalletException {
TransferException(String message): super(message);
}

View file

@ -1,9 +1,6 @@
import 'dart:convert';
import 'dart:io'; import 'dart:io';
import 'package:collection/collection.dart'; import 'package:collection/collection.dart';
import 'package:cw_core/crypto_currency.dart';
import 'package:cw_core/node.dart';
import 'package:cw_core/pathForWallet.dart'; import 'package:cw_core/pathForWallet.dart';
import 'package:cw_core/wallet_base.dart'; import 'package:cw_core/wallet_base.dart';
import 'package:cw_core/wallet_credentials.dart'; import 'package:cw_core/wallet_credentials.dart';
@ -11,17 +8,7 @@ import 'package:cw_core/wallet_info.dart';
import 'package:cw_core/wallet_service.dart'; import 'package:cw_core/wallet_service.dart';
import 'package:cw_core/wallet_type.dart'; import 'package:cw_core/wallet_type.dart';
import 'package:cw_zano/api/api_calls.dart'; import 'package:cw_zano/api/api_calls.dart';
import 'package:cw_zano/api/consts.dart';
import 'package:cw_zano/api/exceptions/already_exists_exception.dart';
import 'package:cw_zano/api/exceptions/create_wallet_exception.dart';
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';
import 'package:cw_zano/model/zano_asset.dart';
import 'package:cw_zano/model/zano_balance.dart';
import 'package:cw_zano/zano_formatter.dart';
import 'package:cw_zano/zano_wallet.dart'; import 'package:cw_zano/zano_wallet.dart';
import 'package:flutter/foundation.dart';
import 'package:hive/hive.dart'; import 'package:hive/hive.dart';
class ZanoNewWalletCredentials extends WalletCredentials { class ZanoNewWalletCredentials extends WalletCredentials {
@ -37,7 +24,13 @@ class ZanoRestoreWalletFromSeedCredentials extends WalletCredentials {
class ZanoRestoreWalletFromKeysCredentials extends WalletCredentials { class ZanoRestoreWalletFromKeysCredentials extends WalletCredentials {
ZanoRestoreWalletFromKeysCredentials( ZanoRestoreWalletFromKeysCredentials(
{required String name, required String password, required this.language, required this.address, required this.viewKey, required this.spendKey, required int height}) {required String name,
required String password,
required this.language,
required this.address,
required this.viewKey,
required this.spendKey,
required int height})
: super(name: name, password: password, height: height); : super(name: name, password: password, height: height);
final String language; final String language;
@ -61,102 +54,25 @@ class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRest
@override @override
Future<ZanoWallet> create(WalletCredentials credentials, {bool? isTestnet}) async { Future<ZanoWallet> create(WalletCredentials credentials, {bool? isTestnet}) async {
print('zanowallet service create isTestnet $isTestnet'); // TODO: remove print('zanowallet service create isTestnet $isTestnet'); // TODO: remove
try { return await ZanoWalletBase.create(credentials: credentials);
final wallet = ZanoWallet(credentials.walletInfo!);
await wallet.connectToNode(node: Node());
final path = await pathForWallet(name: credentials.name, type: getType());
final json = ApiCalls.createWallet(language: '', path: path, password: credentials.password!);
final map = jsonDecode(json) as Map<String, dynamic>;
_checkForCreateWalletError(map);
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
_parseCreateWalletResult(createWalletResult, wallet);
await wallet.store();
await wallet.init(createWalletResult.wi.address);
wallet.addInitialAssets();
return wallet;
} catch (e) {
// TODO: Implement Exception for wallet list service.
print('ZanoWalletsManager Error: ${e.toString()}');
rethrow;
}
} }
@override @override
Future<bool> isWalletExit(String name) async { Future<bool> isWalletExit(String name) async {
try { final path = await pathForWallet(name: name, type: getType());
final path = await pathForWallet(name: name, type: getType()); return ApiCalls.isWalletExist(path: path);
return ApiCalls.isWalletExist(path: path);
} catch (e) {
// TODO: Implement Exception for wallet list service.
print('ZanoWalletsManager Error: $e');
rethrow;
}
} }
@override @override
Future<ZanoWallet> openWallet(String name, String password) async { Future<ZanoWallet> openWallet(String name, String password) async {
final walletInfo = walletInfoSource.values.firstWhereOrNull((info) => info.id == WalletBase.idFor(name, getType()))!;
try { try {
final path = await pathForWallet(name: name, type: getType()); final wallet = await ZanoWalletBase.open(name: name, password: password, walletInfo: walletInfo);
saveBackup(name);
if (walletFilesExist(path)) {
await repairOldAndroidWallet(name);
}
final walletInfo = walletInfoSource.values.firstWhereOrNull((info) => info.id == WalletBase.idFor(name, getType()))!;
final wallet = ZanoWallet(walletInfo);
await wallet.connectToNode(node: Node());
final json = wallet.loadWallet(path, password);
debugPrint('load wallet result $json');
final map = jsonDecode(json) as Map<String, dynamic>;
_checkForCreateWalletError(map);
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
_parseCreateWalletResult(createWalletResult, wallet);
await wallet.store();
await wallet.init(createWalletResult.wi.address);
return wallet; return wallet;
} catch (e) { } catch (e) {
rethrow; await restoreWalletFilesFromBackup(name);
// TODO: uncomment after merge return await ZanoWalletBase.open(name: name, password: password, walletInfo: walletInfo);
//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');
}
if (map['result'] == null) {
throw CreateWalletException('Error creating/loading wallet, empty response');
}
}
void _parseCreateWalletResult(CreateWalletResult result, ZanoWallet wallet) {
hWallet = result.walletId;
wallet.hWallet = hWallet;
wallet.walletAddresses.address = result.wi.address;
for (final item in result.wi.balances) {
if (item.assetInfo.ticker == 'ZANO') {
wallet.balance[CryptoCurrency.zano] = ZanoBalance(
total: item.total,
unlocked: item.unlocked,
decimalPoint: ZanoFormatter.defaultDecimalPoint,
);
} 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,
decimalPoint: asset.decimalPoint,
);
}
}
}
}
if (result.recentHistory.history != null) {
wallet.transfers = result.recentHistory.history!;
} }
} }
@ -190,70 +106,11 @@ class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRest
@override @override
Future<ZanoWallet> restoreFromKeys(ZanoRestoreWalletFromKeysCredentials credentials, {bool? isTestnet}) async { Future<ZanoWallet> restoreFromKeys(ZanoRestoreWalletFromKeysCredentials credentials, {bool? isTestnet}) async {
throw UnimplementedError('Restore from keys not implemented'); throw UnimplementedError();
} }
@override @override
Future<ZanoWallet> restoreFromSeed(ZanoRestoreWalletFromSeedCredentials credentials, {bool? isTestnet}) async { Future<ZanoWallet> restoreFromSeed(ZanoRestoreWalletFromSeedCredentials credentials, {bool? isTestnet}) async {
try { return ZanoWalletBase.restore(credentials: credentials);
final wallet = ZanoWallet(credentials.walletInfo!);
await wallet.connectToNode(node: Node());
final path = await pathForWallet(name: credentials.name, type: getType());
final json = ApiCalls.restoreWalletFromSeed(path: path, password: credentials.password!, seed: credentials.mnemonic);
final map = jsonDecode(json) as Map<String, dynamic>;
if (map['result'] != null) {
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
_parseCreateWalletResult(createWalletResult, wallet);
await wallet.store();
await wallet.init(createWalletResult.wi.address);
wallet.addInitialAssets();
return wallet;
} 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);
}
throw RestoreFromSeedException('', '');
} 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;
}
final oldAndroidWalletDirPath = await outdatedAndroidPathForWalletDir(name: name);
final dir = Directory(oldAndroidWalletDirPath);
if (!dir.existsSync()) {
return;
}
final newWalletDirPath = await pathForWalletDir(name: name, type: getType());
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());
}
} }
} }

View file

@ -15,10 +15,10 @@ class AddressValidator extends TextValidator {
: type == CryptoCurrency.zano && !skipZanoAddressValidation : type == CryptoCurrency.zano && !skipZanoAddressValidation
? ZanoUtils.validateAddress ? ZanoUtils.validateAddress
: null, : null,
pattern: getPattern(type), pattern: getPattern(type, skipZanoAddressValidation),
length: getLength(type)); length: getLength(type));
static String getPattern(CryptoCurrency type) { static String getPattern(CryptoCurrency type, bool skipZanoAddressValidation) {
if (type is Erc20Token) { if (type is Erc20Token) {
return '0x[0-9a-zA-Z]'; return '0x[0-9a-zA-Z]';
} }
@ -126,7 +126,7 @@ class AddressValidator extends TextValidator {
case CryptoCurrency.btcln: case CryptoCurrency.btcln:
return '^(lnbc|LNBC)([0-9]{1,}[a-zA-Z0-9]+)'; return '^(lnbc|LNBC)([0-9]{1,}[a-zA-Z0-9]+)';
case CryptoCurrency.zano: case CryptoCurrency.zano:
return r'$.^'; // always false, we use additional validation then return skipZanoAddressValidation ? '[0-9a-zA-Z]' : r'$.^'; // always false, we use additional validation then
default: default:
return '[0-9a-zA-Z]'; return '[0-9a-zA-Z]';
} }

View file

@ -47,20 +47,6 @@ abstract class HomeSettingsViewModelBase with Store {
@action @action
void setPinNativeToken(bool value) => _settingsStore.pinNativeTokenAtTop = value; void setPinNativeToken(bool value) => _settingsStore.pinNativeTokenAtTop = value;
Future<bool> addAsset(String assetId) async {
if (_balanceViewModel.wallet.type == WalletType.zano) {
try {
final asset = await zano!.addZanoAssetById(_balanceViewModel.wallet, assetId);
_updateTokensList();
_updateFiatPrices(asset);
return true;
} catch (e) {
return false;
}
}
return false;
}
Future<void> addToken(CryptoCurrency token) async { Future<void> addToken(CryptoCurrency token) async {
if (_balanceViewModel.wallet.type == WalletType.ethereum) { if (_balanceViewModel.wallet.type == WalletType.ethereum) {
await ethereum!.addErc20Token(_balanceViewModel.wallet, token); await ethereum!.addErc20Token(_balanceViewModel.wallet, token);

View file

@ -168,7 +168,14 @@ abstract class SendViewModelBase extends WalletChangeListenerViewModel with Stor
PendingTransaction? pendingTransaction; PendingTransaction? pendingTransaction;
@computed @computed
String get balance => wallet.balance[selectedCryptoCurrency]!.formattedAvailableBalance; String get balance {
try {
return wallet.balance[selectedCryptoCurrency]!.formattedAvailableBalance;
} catch (e) {
print(e);
return 'err';
}
}
@computed @computed
bool get isFiatDisabled => balanceViewModel.isFiatDisabled; bool get isFiatDisabled => balanceViewModel.isFiatDisabled;

View file

@ -81,7 +81,7 @@ class CWZano extends Zano {
List<ZanoAsset> getZanoAssets(WalletBase wallet) { List<ZanoAsset> getZanoAssets(WalletBase wallet) {
wallet as ZanoWallet; wallet as ZanoWallet;
return wallet.zanoAssets; return wallet.zanoAssets.values.toList();
} }
@override @override
@ -216,7 +216,7 @@ class CWZano extends Zano {
return CryptoCurrency.zano; return CryptoCurrency.zano;
} }
wallet as ZanoWallet; wallet as ZanoWallet;
return wallet.zanoAssets.firstWhere((element) => element.ticker == transaction.tokenSymbol); return wallet.zanoAssets.values.firstWhere((element) => element.ticker == transaction.tokenSymbol);
} }
String getZanoAssetAddress(CryptoCurrency asset) => (asset as ZanoAsset).assetId; String getZanoAssetAddress(CryptoCurrency asset) => (asset as ZanoAsset).assetId;