mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-10 12:54:38 +00:00
fixed syncing sync status, decimal division, safe null json parsing
This commit is contained in:
parent
75f1f3f7cc
commit
23485a4bab
19 changed files with 232 additions and 151 deletions
|
@ -1,5 +1,8 @@
|
||||||
|
import 'package:decimal/decimal.dart';
|
||||||
|
import 'package:decimal/intl.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:cw_core/crypto_currency.dart';
|
import 'package:cw_core/crypto_currency.dart';
|
||||||
|
import 'package:rational/rational.dart';
|
||||||
|
|
||||||
class AmountConverter {
|
class AmountConverter {
|
||||||
static const _moneroAmountLength = 12;
|
static const _moneroAmountLength = 12;
|
||||||
|
@ -97,7 +100,7 @@ class AmountConverter {
|
||||||
case CryptoCurrency.xusd:
|
case CryptoCurrency.xusd:
|
||||||
return _moneroAmountToString(amount);
|
return _moneroAmountToString(amount);
|
||||||
case CryptoCurrency.zano:
|
case CryptoCurrency.zano:
|
||||||
return _moneroAmountToString(amount);
|
return _moneroAmountToStringUsingDecimals(amount);
|
||||||
default:
|
default:
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
@ -106,9 +109,16 @@ class AmountConverter {
|
||||||
static double cryptoAmountToDouble({required num amount, required num divider}) =>
|
static double cryptoAmountToDouble({required num amount, required num divider}) =>
|
||||||
amount / divider;
|
amount / divider;
|
||||||
|
|
||||||
|
static Decimal cryptoAmountToDecimal({required int amount, required int divider}) =>
|
||||||
|
(Decimal.fromInt(amount) / Decimal.fromInt(divider)).toDecimal();
|
||||||
|
|
||||||
static String _moneroAmountToString(int amount) => _moneroAmountFormat.format(
|
static String _moneroAmountToString(int amount) => _moneroAmountFormat.format(
|
||||||
cryptoAmountToDouble(amount: amount, divider: _moneroAmountDivider));
|
cryptoAmountToDouble(amount: amount, divider: _moneroAmountDivider));
|
||||||
|
|
||||||
|
static String _moneroAmountToStringUsingDecimals(int amount) => _moneroAmountFormat.format(
|
||||||
|
DecimalIntl(cryptoAmountToDecimal(amount: amount, divider: _moneroAmountDivider)));
|
||||||
|
|
||||||
|
|
||||||
static double _moneroAmountToDouble(int amount) =>
|
static double _moneroAmountToDouble(int amount) =>
|
||||||
cryptoAmountToDouble(amount: amount, divider: _moneroAmountDivider);
|
cryptoAmountToDouble(amount: amount, divider: _moneroAmountDivider);
|
||||||
|
|
||||||
|
|
|
@ -177,6 +177,14 @@ packages:
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.2.4"
|
version: "2.2.4"
|
||||||
|
decimal:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: decimal
|
||||||
|
sha256: "24a261d5d5c87e86c7651c417a5dbdf8bcd7080dd592533910e8d0505a279f21"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.3.3"
|
||||||
encrypt:
|
encrypt:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
|
@ -507,6 +515,14 @@ packages:
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.2.1"
|
version: "1.2.1"
|
||||||
|
rational:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: rational
|
||||||
|
sha256: ba58e9e18df9abde280e8b10051e4bce85091e41e8e7e411b6cde2e738d357cf
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.2.2"
|
||||||
shelf:
|
shelf:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|
|
@ -19,6 +19,7 @@ dependencies:
|
||||||
flutter_mobx: ^2.0.6+1
|
flutter_mobx: ^2.0.6+1
|
||||||
intl: ^0.18.0
|
intl: ^0.18.0
|
||||||
encrypt: ^5.0.1
|
encrypt: ^5.0.1
|
||||||
|
decimal: ^2.3.3
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
|
|
|
@ -21,14 +21,14 @@ class AssetInfo {
|
||||||
required this.totalMaxSupply});
|
required this.totalMaxSupply});
|
||||||
|
|
||||||
factory AssetInfo.fromJson(Map<String, dynamic> json) => AssetInfo(
|
factory AssetInfo.fromJson(Map<String, dynamic> json) => AssetInfo(
|
||||||
assetId: json['asset_id'] as String,
|
assetId: json['asset_id'] as String? ?? '',
|
||||||
currentSupply: json['current_supply'] as int,
|
currentSupply: json['current_supply'] as int? ?? 0,
|
||||||
decimalPoint: json['decimal_point'] as int,
|
decimalPoint: json['decimal_point'] as int? ?? 0,
|
||||||
fullName: json['full_name'] as String,
|
fullName: json['full_name'] as String? ?? '',
|
||||||
hiddenSupply: json['hidden_supply'] as bool,
|
hiddenSupply: json['hidden_supply'] as bool,
|
||||||
metaInfo: json['meta_info'] as String,
|
metaInfo: json['meta_info'] as String? ?? '',
|
||||||
owner: json['owner'] as String,
|
owner: json['owner'] as String? ?? '',
|
||||||
ticker: json['ticker'] as String,
|
ticker: json['ticker'] as String? ?? '',
|
||||||
totalMaxSupply: json['total_max_supply'] as int,
|
totalMaxSupply: json['total_max_supply'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,10 +16,10 @@ class Balance {
|
||||||
|
|
||||||
factory Balance.fromJson(Map<String, dynamic> json) => Balance(
|
factory Balance.fromJson(Map<String, dynamic> json) => Balance(
|
||||||
assetInfo:
|
assetInfo:
|
||||||
AssetInfo.fromJson(json['asset_info'] as Map<String, dynamic>),
|
AssetInfo.fromJson(json['asset_info'] as Map<String, dynamic>? ?? {}),
|
||||||
awaitingIn: json['awaiting_in'] as int,
|
awaitingIn: json['awaiting_in'] as int? ?? 0,
|
||||||
awaitingOut: json['awaiting_out'] as int,
|
awaitingOut: json['awaiting_out'] as int? ?? 0,
|
||||||
total: json['total'] as int,
|
total: json['total'] as int? ?? 0,
|
||||||
unlocked: json['unlocked'] as int,
|
unlocked: json['unlocked'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,15 +25,15 @@ class CreateWalletResult {
|
||||||
|
|
||||||
factory CreateWalletResult.fromJson(Map<String, dynamic> json) =>
|
factory CreateWalletResult.fromJson(Map<String, dynamic> json) =>
|
||||||
CreateWalletResult(
|
CreateWalletResult(
|
||||||
name: json['name'] as String,
|
name: json['name'] as String? ?? '',
|
||||||
pass: json['pass'] as String,
|
pass: json['pass'] as String? ?? '',
|
||||||
recentHistory: RecentHistory.fromJson(
|
recentHistory: RecentHistory.fromJson(
|
||||||
json['recent_history'] as Map<String, dynamic>),
|
json['recent_history'] as Map<String, dynamic>? ?? {}),
|
||||||
recovered: json['recovered'] as bool,
|
recovered: json['recovered'] as bool? ?? false,
|
||||||
seed: json['seed'] as String,
|
seed: json['seed'] as String? ?? '',
|
||||||
walletFileSize: json['wallet_file_size'] as int,
|
walletFileSize: json['wallet_file_size'] as int? ?? 0,
|
||||||
walletId: json['wallet_id'] as int,
|
walletId: json['wallet_id'] as int? ?? 0,
|
||||||
walletLocalBcSize: json['wallet_local_bc_size'] as int,
|
walletLocalBcSize: json['wallet_local_bc_size'] as int? ?? 0,
|
||||||
wi: Wi.fromJson(json['wi'] as Map<String, dynamic>),
|
wi: Wi.fromJson(json['wi'] as Map<String, dynamic>? ?? {}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,14 +7,14 @@ class Destination {
|
||||||
{required this.amount, required this.address, required this.assetId});
|
{required this.amount, required this.address, required this.assetId});
|
||||||
|
|
||||||
factory Destination.fromJson(Map<String, dynamic> json) => Destination(
|
factory Destination.fromJson(Map<String, dynamic> json) => Destination(
|
||||||
amount: int.parse(json['amount'] as String),
|
amount: int.parse(json['amount'] as String? ?? '0'),
|
||||||
address: json['address'] as String,
|
address: json['address'] as String? ?? '',
|
||||||
assetId: json['asset_id'] as String,
|
assetId: json['asset_id'] as String? ?? '',
|
||||||
);
|
);
|
||||||
|
|
||||||
Map<String, dynamic> toJson() => {
|
Map<String, dynamic> toJson() => {
|
||||||
"amount": amount.toString(),
|
'amount': amount.toString(),
|
||||||
"address": address,
|
'address': address,
|
||||||
"asset_id": assetId,
|
'asset_id': assetId,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,9 +8,9 @@ class GetAddressInfoResult {
|
||||||
{required this.valid, required this.auditable, required this.paymentId, required this.wrap});
|
{required this.valid, required this.auditable, required this.paymentId, required this.wrap});
|
||||||
|
|
||||||
factory GetAddressInfoResult.fromJson(Map<String, dynamic> json) => GetAddressInfoResult(
|
factory GetAddressInfoResult.fromJson(Map<String, dynamic> json) => GetAddressInfoResult(
|
||||||
valid: json['valid'] as bool,
|
valid: json['valid'] as bool? ?? false,
|
||||||
auditable: json['auditable'] as bool,
|
auditable: json['auditable'] as bool? ?? false,
|
||||||
paymentId: json['payment_id'] as bool,
|
paymentId: json['payment_id'] as bool? ?? false,
|
||||||
wrap: json['wrap'] as bool,
|
wrap: json['wrap'] as bool? ?? false,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,27 +43,27 @@ class History {
|
||||||
});
|
});
|
||||||
|
|
||||||
factory History.fromJson(Map<String, dynamic> json) => History(
|
factory History.fromJson(Map<String, dynamic> json) => History(
|
||||||
comment: json['comment'] as String,
|
comment: json['comment'] as String? ?? '',
|
||||||
employedEntries: EmployedEntries.fromJson(
|
employedEntries: EmployedEntries.fromJson(
|
||||||
json['employed_entries'] as Map<String, dynamic>),
|
json['employed_entries'] as Map<String, dynamic>? ?? {}),
|
||||||
fee: json['fee'] as int,
|
fee: json['fee'] as int? ?? 0,
|
||||||
height: json['height'] as int,
|
height: json['height'] as int? ?? 0,
|
||||||
isMining: json['is_mining'] as bool,
|
isMining: json['is_mining'] as bool? ?? false,
|
||||||
isMixing: json['is_mixing'] as bool,
|
isMixing: json['is_mixing'] as bool? ?? false,
|
||||||
isService: json['is_service'] as bool,
|
isService: json['is_service'] as bool? ?? false,
|
||||||
paymentId: json['payment_id'] as String,
|
paymentId: json['payment_id'] as String? ?? '',
|
||||||
remoteAddresses: json['remote_addresses'] == null ? [] :
|
remoteAddresses: json['remote_addresses'] == null ? [] :
|
||||||
(json['remote_addresses'] as List<dynamic>).cast<String>(),
|
(json['remote_addresses'] as List<dynamic>).cast<String>(),
|
||||||
remoteAliases: json['remote_aliases'] == null ? [] : (json['remote_aliases'] as List<dynamic>).cast<String>(),
|
remoteAliases: json['remote_aliases'] == null ? [] : (json['remote_aliases'] as List<dynamic>).cast<String>(),
|
||||||
showSender: json['show_sender'] as bool,
|
showSender: json['show_sender'] as bool? ?? false,
|
||||||
subtransfers: (json['subtransfers'] as List<dynamic>)
|
subtransfers: (json['subtransfers'] as List<dynamic>? ?? [])
|
||||||
.map((e) => Subtransfer.fromJson(e as Map<String, dynamic>))
|
.map((e) => Subtransfer.fromJson(e as Map<String, dynamic>))
|
||||||
.toList(),
|
.toList(),
|
||||||
timestamp: json['timestamp'] as int,
|
timestamp: json['timestamp'] as int? ?? 0,
|
||||||
transferInternalIndex: json['transfer_internal_index'] is double ? (json['transfer_internal_index'] as double).toInt() : json['transfer_internal_index'] as int,
|
transferInternalIndex: json['transfer_internal_index'] == null ? 0 : json['transfer_internal_index'] is double ? (json['transfer_internal_index'] as double).toInt() : json['transfer_internal_index'] as int,
|
||||||
txBlobSize: json['tx_blob_size'] as int,
|
txBlobSize: json['tx_blob_size'] as int? ?? 0,
|
||||||
txHash: json['tx_hash'] as String,
|
txHash: json['tx_hash'] as String? ?? '',
|
||||||
txType: json['tx_type'] as int,
|
txType: json['tx_type'] as int? ?? 0,
|
||||||
unlockTime: json['unlock_time'] as int,
|
unlockTime: json['unlock_time'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,8 +6,8 @@ class Receive {
|
||||||
Receive({required this.amount, required this.assetId, required this.index});
|
Receive({required this.amount, required this.assetId, required this.index});
|
||||||
|
|
||||||
factory Receive.fromJson(Map<String, dynamic> json) => Receive(
|
factory Receive.fromJson(Map<String, dynamic> json) => Receive(
|
||||||
amount: json['amount'] as int,
|
amount: json['amount'] as int? ?? 0,
|
||||||
assetId: json['asset_id'] as String,
|
assetId: json['asset_id'] as String? ?? '',
|
||||||
index: json['index'] as int,
|
index: json['index'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,7 +14,7 @@ class RecentHistory {
|
||||||
history: json['history'] == null ? null : (json['history'] as List<dynamic>)
|
history: json['history'] == null ? null : (json['history'] as List<dynamic>)
|
||||||
.map((e) => History.fromJson(e as Map<String, dynamic>))
|
.map((e) => History.fromJson(e as Map<String, dynamic>))
|
||||||
.toList(),
|
.toList(),
|
||||||
lastItemIndex: json['last_item_index'] as int,
|
lastItemIndex: json['last_item_index'] as int? ?? 0,
|
||||||
totalHistoryItems: json['total_history_items'] as int,
|
totalHistoryItems: json['total_history_items'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,6 @@ class StoreResult {
|
||||||
StoreResult({required this.walletFileSize});
|
StoreResult({required this.walletFileSize});
|
||||||
|
|
||||||
factory StoreResult.fromJson(Map<String, dynamic> json) => StoreResult(
|
factory StoreResult.fromJson(Map<String, dynamic> json) => StoreResult(
|
||||||
walletFileSize: json['wallet_file_size'] as int,
|
walletFileSize: json['wallet_file_size'] as int? ?? 0,
|
||||||
);
|
);
|
||||||
}
|
}
|
|
@ -7,8 +7,8 @@ class Subtransfer {
|
||||||
{required this.amount, required this.assetId, required this.isIncome});
|
{required this.amount, required this.assetId, required this.isIncome});
|
||||||
|
|
||||||
factory Subtransfer.fromJson(Map<String, dynamic> json) => Subtransfer(
|
factory Subtransfer.fromJson(Map<String, dynamic> json) => Subtransfer(
|
||||||
amount: json['amount'] as int,
|
amount: json['amount'] as int? ?? 0,
|
||||||
assetId: json['asset_id'] as String,
|
assetId: json['asset_id'] as String? ?? '',
|
||||||
isIncome: json['is_income'] as bool,
|
isIncome: json['is_income'] as bool? ?? false,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,13 +20,13 @@ class TransferParams {
|
||||||
});
|
});
|
||||||
|
|
||||||
Map<String, dynamic> toJson() => {
|
Map<String, dynamic> toJson() => {
|
||||||
"destinations": destinations,
|
'destinations': destinations,
|
||||||
"fee": fee,
|
'fee': fee,
|
||||||
"mixin": mixin,
|
'mixin': mixin,
|
||||||
"payment_id": paymentId,
|
'payment_id': paymentId,
|
||||||
"comment": comment,
|
'comment': comment,
|
||||||
"push_payer": pushPayer,
|
'push_payer': pushPayer,
|
||||||
"hide_receiver": hideReceiver,
|
'hide_receiver': hideReceiver,
|
||||||
};
|
};
|
||||||
|
|
||||||
factory TransferParams.fromJson(Map<String, dynamic> json) => TransferParams(
|
factory TransferParams.fromJson(Map<String, dynamic> json) => TransferParams(
|
||||||
|
@ -35,7 +35,7 @@ class TransferParams {
|
||||||
mixin: json['mixin'] as int? ?? 0,
|
mixin: json['mixin'] as int? ?? 0,
|
||||||
paymentId: json['payment_id'] as String? ?? '',
|
paymentId: json['payment_id'] as String? ?? '',
|
||||||
comment: json['comment'] as String? ?? '',
|
comment: json['comment'] as String? ?? '',
|
||||||
pushPayer: json["push_payer"] as bool? ?? false,
|
pushPayer: json['push_payer'] as bool? ?? false,
|
||||||
hideReceiver: json["hide_receiver"] as bool? ?? false,
|
hideReceiver: json['hide_receiver'] as bool? ?? false,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,10 +8,10 @@ class WiExtended {
|
||||||
WiExtended({required this.seed, required this.spendPrivateKey, required this.spendPublicKey, required this.viewPrivateKey, required this.viewPublicKey});
|
WiExtended({required this.seed, required this.spendPrivateKey, required this.spendPublicKey, required this.viewPrivateKey, required this.viewPublicKey});
|
||||||
|
|
||||||
factory WiExtended.fromJson(Map<String, dynamic> json) => WiExtended(
|
factory WiExtended.fromJson(Map<String, dynamic> json) => WiExtended(
|
||||||
seed: json["seed"] as String? ?? '',
|
seed: json['seed'] as String? ?? '',
|
||||||
spendPrivateKey: json["spend_private_key"] as String? ?? '',
|
spendPrivateKey: json['spend_private_key'] as String? ?? '',
|
||||||
spendPublicKey: json["spend_public_key"] as String? ?? '',
|
spendPublicKey: json['spend_public_key'] as String? ?? '',
|
||||||
viewPrivateKey: json["view_private_key"] as String? ?? '',
|
viewPrivateKey: json['view_private_key'] as String? ?? '',
|
||||||
viewPublicKey: json["view_public_key"] as String? ?? '',
|
viewPublicKey: json['view_public_key'] as String? ?? '',
|
||||||
);
|
);
|
||||||
}
|
}
|
|
@ -2,5 +2,5 @@ import 'dart:ffi';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
|
|
||||||
final DynamicLibrary zanoApi = Platform.isAndroid
|
final DynamicLibrary zanoApi = Platform.isAndroid
|
||||||
? DynamicLibrary.open("libcw_zano.so")
|
? DynamicLibrary.open('libcw_zano.so')
|
||||||
: DynamicLibrary.open("cw_zano.framework/cw_zano");
|
: DynamicLibrary.open('cw_zano.framework/cw_zano');
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
|
import 'package:cw_core/amount_converter.dart';
|
||||||
import 'package:cw_core/balance.dart';
|
import 'package:cw_core/balance.dart';
|
||||||
|
import 'package:cw_core/crypto_currency.dart';
|
||||||
import 'package:cw_core/monero_amount_format.dart';
|
import 'package:cw_core/monero_amount_format.dart';
|
||||||
|
|
||||||
class ZanoBalance extends Balance {
|
class ZanoBalance extends Balance {
|
||||||
|
@ -7,10 +9,10 @@ class ZanoBalance extends Balance {
|
||||||
ZanoBalance({required this.total, required this.unlocked}): super(unlocked, total-unlocked);
|
ZanoBalance({required this.total, required this.unlocked}): super(unlocked, total-unlocked);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get formattedAdditionalBalance => moneroAmountToString(amount: total-unlocked);
|
String get formattedAdditionalBalance => AmountConverter.amountIntToString(CryptoCurrency.zano, total-unlocked);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get formattedAvailableBalance => moneroAmountToString(amount: unlocked);
|
String get formattedAvailableBalance => AmountConverter.amountIntToString(CryptoCurrency.zano, unlocked);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get formattedFrozenBalance => '';
|
String get formattedFrozenBalance => '';
|
||||||
|
|
|
@ -40,28 +40,32 @@ const moneroBlockSize = 1000;
|
||||||
|
|
||||||
class ZanoWallet = ZanoWalletBase with _$ZanoWallet;
|
class ZanoWallet = ZanoWalletBase with _$ZanoWallet;
|
||||||
|
|
||||||
typedef _load_wallet = Pointer<Utf8> Function(Pointer<Utf8>, Pointer<Utf8>, Int8);
|
typedef _load_wallet = Pointer<Utf8> Function(
|
||||||
|
Pointer<Utf8>, Pointer<Utf8>, Int8);
|
||||||
typedef _LoadWallet = Pointer<Utf8> Function(Pointer<Utf8>, Pointer<Utf8>, int);
|
typedef _LoadWallet = Pointer<Utf8> Function(Pointer<Utf8>, Pointer<Utf8>, int);
|
||||||
|
|
||||||
const int zanoMixin = 10;
|
const int zanoMixin = 10;
|
||||||
|
|
||||||
abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHistory, ZanoTransactionInfo> with Store {
|
abstract class ZanoWalletBase
|
||||||
|
extends WalletBase<ZanoBalance, ZanoTransactionHistory, ZanoTransactionInfo>
|
||||||
|
with Store {
|
||||||
ZanoWalletBase(WalletInfo walletInfo)
|
ZanoWalletBase(WalletInfo walletInfo)
|
||||||
: balance = ObservableMap.of({CryptoCurrency.zano: ZanoBalance(total: 0, unlocked: 0)}),
|
: balance = ObservableMap.of(
|
||||||
|
{CryptoCurrency.zano: ZanoBalance(total: 0, unlocked: 0)}),
|
||||||
_isTransactionUpdating = false,
|
_isTransactionUpdating = false,
|
||||||
_hasSyncAfterStartup = false,
|
_hasSyncAfterStartup = false,
|
||||||
walletAddresses = ZanoWalletAddresses(walletInfo),
|
walletAddresses = ZanoWalletAddresses(walletInfo),
|
||||||
syncStatus = NotConnectedSyncStatus(),
|
syncStatus = NotConnectedSyncStatus(),
|
||||||
super(walletInfo) {
|
super(walletInfo) {
|
||||||
transactionHistory = ZanoTransactionHistory();
|
transactionHistory = ZanoTransactionHistory();
|
||||||
/*_onAccountChangeReaction =
|
// _onAccountChangeReaction =
|
||||||
reaction((_) => walletAddresses.account, (Account? account) {
|
// reaction((_) => walletAddresses.account, (Account? account) {
|
||||||
if (account == null) {
|
// if (account == null) {
|
||||||
return;
|
// return;
|
||||||
}
|
// }
|
||||||
balance.addAll(getZanoBalance(accountIndex: account.id));
|
// balance.addAll(getZanoBalance(accountIndex: account.id));
|
||||||
/**walletAddresses.updateSubaddressList(accountIndex: account.id);*/
|
// /**walletAddresses.updateSubaddressList(accountIndex: account.id);*/
|
||||||
});*/
|
// });
|
||||||
}
|
}
|
||||||
|
|
||||||
List<History> history = [];
|
List<History> history = [];
|
||||||
|
@ -86,10 +90,14 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
String seed = '';
|
String seed = '';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
ZanoWalletKeys keys = ZanoWalletKeys(privateSpendKey: '', privateViewKey: '', publicSpendKey: '', publicViewKey: '');
|
ZanoWalletKeys keys = ZanoWalletKeys(
|
||||||
|
privateSpendKey: '',
|
||||||
|
privateViewKey: '',
|
||||||
|
publicSpendKey: '',
|
||||||
|
publicViewKey: '');
|
||||||
|
|
||||||
//zano_wallet.SyncListener? _listener;
|
//zano_wallet.SyncListener? _listener;
|
||||||
/**ReactionDisposer? _onAccountChangeReaction;*/
|
// ReactionDisposer? _onAccountChangeReaction;
|
||||||
Timer? _updateSyncInfoTimer;
|
Timer? _updateSyncInfoTimer;
|
||||||
int _cachedBlockchainHeight = 0;
|
int _cachedBlockchainHeight = 0;
|
||||||
int _lastKnownBlockHeight = 0;
|
int _lastKnownBlockHeight = 0;
|
||||||
|
@ -114,7 +122,8 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
//_setListeners();
|
//_setListeners();
|
||||||
await updateTransactions();
|
await updateTransactions();
|
||||||
|
|
||||||
_autoSaveTimer = Timer.periodic(Duration(seconds: _autoSaveInterval), (_) async => await save());
|
_autoSaveTimer = Timer.periodic(
|
||||||
|
Duration(seconds: _autoSaveInterval), (_) async => await save());
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
@ -122,9 +131,10 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void close() {
|
void close() {
|
||||||
|
ApiCalls.closeWallet(hWallet: hWallet);
|
||||||
_updateSyncInfoTimer?.cancel();
|
_updateSyncInfoTimer?.cancel();
|
||||||
//_listener?.stop();
|
//_listener?.stop();
|
||||||
/**_onAccountChangeReaction?.reaction.dispose();*/
|
// _onAccountChangeReaction?.reaction.dispose();
|
||||||
_autoSaveTimer?.cancel();
|
_autoSaveTimer?.cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -133,9 +143,9 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
try {
|
try {
|
||||||
syncStatus = ConnectingSyncStatus();
|
syncStatus = ConnectingSyncStatus();
|
||||||
await ApiCalls.setupNode(
|
await ApiCalls.setupNode(
|
||||||
address: "195.201.107.230:33336", // node.uriRaw,
|
address: '195.201.107.230:33336', // node.uriRaw,
|
||||||
login: "", // node.login,
|
login: '', // node.login,
|
||||||
password: "", // node.password,
|
password: '', // node.password,
|
||||||
useSSL: false, // node.useSSL ?? false,
|
useSSL: false, // node.useSSL ?? false,
|
||||||
isLightWallet: false, // FIXME: hardcoded value
|
isLightWallet: false, // FIXME: hardcoded value
|
||||||
/*socksProxyAddress: node.socksProxyAddress*/
|
/*socksProxyAddress: node.socksProxyAddress*/
|
||||||
|
@ -149,6 +159,31 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void _updateSyncProgress(GetWalletStatusResult walletStatus) {
|
||||||
|
final syncHeight = walletStatus.currentWalletHeight;
|
||||||
|
if (_initialSyncHeight <= 0) {
|
||||||
|
_initialSyncHeight = syncHeight;
|
||||||
|
}
|
||||||
|
final bchHeight = walletStatus.currentDaemonHeight;
|
||||||
|
|
||||||
|
if (_lastKnownBlockHeight == syncHeight) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_lastKnownBlockHeight = syncHeight;
|
||||||
|
final track = bchHeight - _initialSyncHeight;
|
||||||
|
final diff = track - (bchHeight - syncHeight);
|
||||||
|
final ptc = diff <= 0 ? 0.0 : diff / track;
|
||||||
|
final left = bchHeight - syncHeight;
|
||||||
|
|
||||||
|
if (syncHeight < 0 || left < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 1. Actual new height; 2. Blocks left to finish; 3. Progress in percents;
|
||||||
|
_onNewBlock.call(syncHeight, left, ptc);
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Future<void> startSync() async {
|
Future<void> startSync() async {
|
||||||
try {
|
try {
|
||||||
|
@ -156,51 +191,31 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
_cachedBlockchainHeight = 0;
|
_cachedBlockchainHeight = 0;
|
||||||
_lastKnownBlockHeight = 0;
|
_lastKnownBlockHeight = 0;
|
||||||
_initialSyncHeight = 0;
|
_initialSyncHeight = 0;
|
||||||
_updateSyncInfoTimer ??= Timer.periodic(Duration(milliseconds: 1200), (_) async {
|
_updateSyncInfoTimer ??=
|
||||||
|
Timer.periodic(Duration(milliseconds: 1200), (_) async {
|
||||||
/**if (isNewTransactionExist()) {
|
/**if (isNewTransactionExist()) {
|
||||||
onNewTransaction?.call();
|
onNewTransaction?.call();
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
GetWalletStatusResult status = getWalletStatus();
|
final walletStatus = getWalletStatus();
|
||||||
|
_updateSyncProgress(walletStatus);
|
||||||
// You can call getWalletInfo ONLY if getWalletStatus returns NOT is in long refresh and wallet state is 2 (ready)
|
// You can call getWalletInfo ONLY if getWalletStatus returns NOT is in long refresh and wallet state is 2 (ready)
|
||||||
if (!status.isInLongRefresh && status.walletState == 2) {
|
if (!walletStatus.isInLongRefresh && walletStatus.walletState == 2) {
|
||||||
final syncHeight = status.currentWalletHeight;
|
final walletInfo = getWalletInfo();
|
||||||
|
seed = walletInfo.wiExtended.seed;
|
||||||
GetWalletInfoResult result = getWalletInfo();
|
|
||||||
seed = result.wiExtended.seed;
|
|
||||||
keys = ZanoWalletKeys(
|
keys = ZanoWalletKeys(
|
||||||
privateSpendKey: result.wiExtended.spendPrivateKey,
|
privateSpendKey: walletInfo.wiExtended.spendPrivateKey,
|
||||||
privateViewKey: result.wiExtended.viewPrivateKey,
|
privateViewKey: walletInfo.wiExtended.viewPrivateKey,
|
||||||
publicSpendKey: result.wiExtended.spendPublicKey,
|
publicSpendKey: walletInfo.wiExtended.spendPublicKey,
|
||||||
publicViewKey: result.wiExtended.viewPublicKey,
|
publicViewKey: walletInfo.wiExtended.viewPublicKey,
|
||||||
);
|
);
|
||||||
|
|
||||||
final _balance = result.wi.balances.first;
|
final _balance = walletInfo.wi.balances.first;
|
||||||
defaultAsssetId = _balance.assetInfo.assetId;
|
defaultAsssetId = _balance.assetInfo.assetId;
|
||||||
balance = ObservableMap.of({CryptoCurrency.zano: ZanoBalance(total: _balance.total, unlocked: _balance.unlocked)});
|
balance = ObservableMap.of({
|
||||||
|
CryptoCurrency.zano:
|
||||||
if (_initialSyncHeight <= 0) {
|
ZanoBalance(total: _balance.total, unlocked: _balance.unlocked)
|
||||||
_initialSyncHeight = syncHeight;
|
});
|
||||||
}
|
|
||||||
|
|
||||||
final bchHeight = status.currentDaemonHeight;
|
|
||||||
|
|
||||||
if (_lastKnownBlockHeight == syncHeight) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_lastKnownBlockHeight = syncHeight;
|
|
||||||
final track = bchHeight - _initialSyncHeight;
|
|
||||||
final diff = track - (bchHeight - syncHeight);
|
|
||||||
final ptc = diff <= 0 ? 0.0 : diff / track;
|
|
||||||
final left = bchHeight - syncHeight;
|
|
||||||
|
|
||||||
if (syncHeight < 0 || left < 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 1. Actual new height; 2. Blocks left to finish; 3. Progress in percents;
|
|
||||||
_onNewBlock.call(syncHeight, left, ptc);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
@ -219,10 +234,12 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
final fee = calculateEstimatedFee(creds.priority);
|
final fee = calculateEstimatedFee(creds.priority);
|
||||||
late List<Destination> destinations;
|
late List<Destination> destinations;
|
||||||
if (hasMultiDestination) {
|
if (hasMultiDestination) {
|
||||||
if (outputs.any((output) => output.sendAll || (output.formattedCryptoAmount ?? 0) <= 0)) {
|
if (outputs.any((output) =>
|
||||||
|
output.sendAll || (output.formattedCryptoAmount ?? 0) <= 0)) {
|
||||||
throw ZanoTransactionCreationException("You don't have enough coins.");
|
throw ZanoTransactionCreationException("You don't have enough coins.");
|
||||||
}
|
}
|
||||||
final int totalAmount = outputs.fold(0, (acc, value) => acc + (value.formattedCryptoAmount ?? 0));
|
final int totalAmount = outputs.fold(
|
||||||
|
0, (acc, value) => acc + (value.formattedCryptoAmount ?? 0));
|
||||||
if (totalAmount + fee > unlockedBalance) {
|
if (totalAmount + fee > unlockedBalance) {
|
||||||
throw ZanoTransactionCreationException(
|
throw ZanoTransactionCreationException(
|
||||||
"You don't have enough coins (required: ${moneroAmountToString(amount: totalAmount + fee)}, unlocked ${moneroAmountToString(amount: unlockedBalance)}).");
|
"You don't have enough coins (required: ${moneroAmountToString(amount: totalAmount + fee)}, unlocked ${moneroAmountToString(amount: unlockedBalance)}).");
|
||||||
|
@ -230,7 +247,9 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
destinations = outputs
|
destinations = outputs
|
||||||
.map((output) => Destination(
|
.map((output) => Destination(
|
||||||
amount: output.formattedCryptoAmount ?? 0,
|
amount: output.formattedCryptoAmount ?? 0,
|
||||||
address: output.isParsedAddress ? output.extractedAddress! : output.address,
|
address: output.isParsedAddress
|
||||||
|
? output.extractedAddress!
|
||||||
|
: output.address,
|
||||||
assetId: defaultAsssetId,
|
assetId: defaultAsssetId,
|
||||||
))
|
))
|
||||||
.toList();
|
.toList();
|
||||||
|
@ -249,13 +268,16 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
destinations = [
|
destinations = [
|
||||||
Destination(
|
Destination(
|
||||||
amount: amount,
|
amount: amount,
|
||||||
address: output.isParsedAddress ? output.extractedAddress! : output.address,
|
address: output.isParsedAddress
|
||||||
|
? output.extractedAddress!
|
||||||
|
: output.address,
|
||||||
assetId: defaultAsssetId,
|
assetId: defaultAsssetId,
|
||||||
)
|
)
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
destinations.forEach((destination) {
|
destinations.forEach((destination) {
|
||||||
debugPrint('destination ${destination.address} ${destination.amount} ${destination.assetId}');
|
debugPrint(
|
||||||
|
'destination ${destination.address} ${destination.amount} ${destination.assetId}');
|
||||||
});
|
});
|
||||||
return PendingZanoTransaction(
|
return PendingZanoTransaction(
|
||||||
zanoWallet: this,
|
zanoWallet: this,
|
||||||
|
@ -266,7 +288,8 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
int calculateEstimatedFee(TransactionPriority priority, [int? amount = null]) {
|
int calculateEstimatedFee(TransactionPriority priority,
|
||||||
|
[int? amount = null]) {
|
||||||
return ApiCalls.getCurrentTxFee(priority: priority.raw);
|
return ApiCalls.getCurrentTxFee(priority: priority.raw);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -288,7 +311,8 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
if (map['result'] == null || map['result']['result'] == null) {
|
if (map['result'] == null || map['result']['result'] == null) {
|
||||||
throw 'store empty response';
|
throw 'store empty response';
|
||||||
}
|
}
|
||||||
final _ = StoreResult.fromJson(map['result']['result'] as Map<String, dynamic>);
|
final _ =
|
||||||
|
StoreResult.fromJson(map['result']['result'] as Map<String, dynamic>);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print(e.toString());
|
print(e.toString());
|
||||||
}
|
}
|
||||||
|
@ -344,7 +368,8 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
|
|
||||||
Future<void> _refreshTransactions() async {
|
Future<void> _refreshTransactions() async {
|
||||||
try {
|
try {
|
||||||
final result = await invokeMethod('get_recent_txs_and_info', GetRecentTxsAndInfoParams(offset: 0, count: 30));
|
final result = await invokeMethod('get_recent_txs_and_info',
|
||||||
|
GetRecentTxsAndInfoParams(offset: 0, count: 30));
|
||||||
final map = jsonDecode(result) as Map<String, dynamic>?;
|
final map = jsonDecode(result) as Map<String, dynamic>?;
|
||||||
if (map == null) {
|
if (map == null) {
|
||||||
print('get_recent_txs_and_info empty response');
|
print('get_recent_txs_and_info empty response');
|
||||||
|
@ -368,7 +393,9 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
history = transfers.map((e) => History.fromJson(e as Map<String, dynamic>)).toList();
|
history = transfers
|
||||||
|
.map((e) => History.fromJson(e as Map<String, dynamic>))
|
||||||
|
.toList();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print(e.toString());
|
print(e.toString());
|
||||||
}
|
}
|
||||||
|
@ -378,7 +405,10 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
Future<Map<String, ZanoTransactionInfo>> fetchTransactions() async {
|
Future<Map<String, ZanoTransactionInfo>> fetchTransactions() async {
|
||||||
try {
|
try {
|
||||||
await _refreshTransactions();
|
await _refreshTransactions();
|
||||||
return history.map<ZanoTransactionInfo>((history) => ZanoTransactionInfo.fromHistory(history)).fold<Map<String, ZanoTransactionInfo>>(
|
return history
|
||||||
|
.map<ZanoTransactionInfo>(
|
||||||
|
(history) => ZanoTransactionInfo.fromHistory(history))
|
||||||
|
.fold<Map<String, ZanoTransactionInfo>>(
|
||||||
<String, ZanoTransactionInfo>{},
|
<String, ZanoTransactionInfo>{},
|
||||||
(Map<String, ZanoTransactionInfo> acc, ZanoTransactionInfo tx) {
|
(Map<String, ZanoTransactionInfo> acc, ZanoTransactionInfo tx) {
|
||||||
acc[tx.id] = tx;
|
acc[tx.id] = tx;
|
||||||
|
@ -420,10 +450,12 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
// }
|
// }
|
||||||
|
|
||||||
void _askForUpdateBalance() {
|
void _askForUpdateBalance() {
|
||||||
debugPrint('askForUpdateBalance'); // TODO: remove, also remove this method completely
|
debugPrint(
|
||||||
|
'askForUpdateBalance'); // TODO: remove, also remove this method completely
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _askForUpdateTransactionHistory() async => await updateTransactions();
|
Future<void> _askForUpdateTransactionHistory() async =>
|
||||||
|
await updateTransactions();
|
||||||
|
|
||||||
void _onNewBlock(int height, int blocksLeft, double ptc) async {
|
void _onNewBlock(int height, int blocksLeft, double ptc) async {
|
||||||
try {
|
try {
|
||||||
|
@ -473,7 +505,10 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<String> invokeMethod(String methodName, Object params) async {
|
Future<String> invokeMethod(String methodName, Object params) async {
|
||||||
var invokeResult = ApiCalls.asyncCall(methodName: 'invoke', hWallet: hWallet, params: '{"method": "$methodName","params": ${jsonEncode(params)}}');
|
var invokeResult = 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>;
|
||||||
int attempts = 0;
|
int attempts = 0;
|
||||||
if (map['job_id'] != null) {
|
if (map['job_id'] != null) {
|
||||||
|
@ -482,7 +517,9 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
await Future.delayed(Duration(milliseconds: attempts < 2 ? 100 : 500));
|
await Future.delayed(Duration(milliseconds: attempts < 2 ? 100 : 500));
|
||||||
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) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
} while (++attempts < _maxAttempts);
|
} while (++attempts < _maxAttempts);
|
||||||
|
@ -493,14 +530,16 @@ abstract class ZanoWalletBase extends WalletBase<ZanoBalance, ZanoTransactionHis
|
||||||
GetWalletInfoResult getWalletInfo() {
|
GetWalletInfoResult getWalletInfo() {
|
||||||
final json = ApiCalls.getWalletInfo(hWallet);
|
final json = ApiCalls.getWalletInfo(hWallet);
|
||||||
print('wallet info $json'); // TODO: remove
|
print('wallet info $json'); // TODO: remove
|
||||||
final result = GetWalletInfoResult.fromJson(jsonDecode(json) as Map<String, dynamic>);
|
final result =
|
||||||
|
GetWalletInfoResult.fromJson(jsonDecode(json) as Map<String, dynamic>);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
GetWalletStatusResult getWalletStatus() {
|
GetWalletStatusResult getWalletStatus() {
|
||||||
final json = ApiCalls.getWalletStatus(hWallet: hWallet);
|
final json = ApiCalls.getWalletStatus(hWallet: hWallet);
|
||||||
print('wallet status $json'); // TODO: remove
|
print('wallet status $json'); // TODO: remove
|
||||||
final status = GetWalletStatusResult.fromJson(jsonDecode(json) as Map<String, dynamic>);
|
final status = GetWalletStatusResult.fromJson(
|
||||||
|
jsonDecode(json) as Map<String, dynamic>);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -63,9 +63,9 @@ class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRest
|
||||||
final wallet = ZanoWallet(credentials.walletInfo!);
|
final wallet = ZanoWallet(credentials.walletInfo!);
|
||||||
await wallet.connectToNode(node: Node());
|
await wallet.connectToNode(node: Node());
|
||||||
final path = await pathForWallet(name: credentials.name, type: getType());
|
final path = await pathForWallet(name: credentials.name, type: getType());
|
||||||
final result = ApiCalls.createWallet(language: "", path: path, password: credentials.password!);
|
final result = ApiCalls.createWallet(language: '', path: path, password: credentials.password!);
|
||||||
final map = json.decode(result) as Map<String, dynamic>;
|
final map = json.decode(result) as Map<String, dynamic>;
|
||||||
if (map['result'] == null) throw CreateWalletException('');
|
_checkForCreateWalletError(map);
|
||||||
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
|
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
|
||||||
_parseCreateWalletResult(createWalletResult, wallet);
|
_parseCreateWalletResult(createWalletResult, wallet);
|
||||||
await wallet.store();
|
await wallet.store();
|
||||||
|
@ -103,9 +103,9 @@ class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRest
|
||||||
final wallet = ZanoWallet(walletInfo);
|
final wallet = ZanoWallet(walletInfo);
|
||||||
await wallet.connectToNode(node: Node());
|
await wallet.connectToNode(node: Node());
|
||||||
final result = wallet.loadWallet(path, password);
|
final result = wallet.loadWallet(path, password);
|
||||||
print("load wallet result $result");
|
print('load wallet result $result');
|
||||||
final map = json.decode(result) as Map<String, dynamic>;
|
final map = json.decode(result) as Map<String, dynamic>;
|
||||||
if (map['result'] == null) throw CreateWalletException('');
|
_checkForCreateWalletError(map);
|
||||||
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
|
final createWalletResult = CreateWalletResult.fromJson(map['result'] as Map<String, dynamic>);
|
||||||
_parseCreateWalletResult(createWalletResult, wallet);
|
_parseCreateWalletResult(createWalletResult, wallet);
|
||||||
await wallet.store();
|
await wallet.store();
|
||||||
|
@ -113,6 +113,19 @@ class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRest
|
||||||
return wallet;
|
return wallet;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
rethrow;
|
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');
|
||||||
|
}
|
||||||
|
if (map['result'] == null) {
|
||||||
|
throw CreateWalletException('Error creating/loading wallet, empty response');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -158,7 +171,7 @@ class ZanoWalletService extends WalletService<ZanoNewWalletCredentials, ZanoRest
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Future<ZanoWallet> restoreFromKeys(ZanoRestoreWalletFromKeysCredentials credentials) async {
|
Future<ZanoWallet> restoreFromKeys(ZanoRestoreWalletFromKeysCredentials credentials) async {
|
||||||
throw UnimplementedError("Restore from keys not implemented");
|
throw UnimplementedError('Restore from keys not implemented');
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
|
Loading…
Reference in a new issue