cake_wallet/cw_monero/lib/api/wallet.dart

251 lines
7.1 KiB
Dart
Raw Normal View History

import 'dart:async';
2024-04-12 12:54:24 +00:00
2024-04-10 12:27:10 +00:00
import 'package:cw_monero/api/account_list.dart';
import 'package:cw_monero/api/exceptions/setup_wallet_exception.dart';
2024-04-10 12:27:10 +00:00
import 'package:monero/monero.dart' as monero;
int _boolToInt(bool value) => value ? 1 : 0;
2024-04-16 15:15:20 +00:00
int getSyncingHeight() => monero.MONERO_cw_WalletListener_height(getWlptr());
bool isNeededToRefresh() {
final ret = monero.MONERO_cw_WalletListener_isNeedToRefresh(getWlptr());
monero.MONERO_cw_WalletListener_resetNeedToRefresh(getWlptr());
return ret;
}
bool isNewTransactionExist() {
final ret = monero.MONERO_cw_WalletListener_isNewTransactionExist(getWlptr());
monero.MONERO_cw_WalletListener_resetIsNewTransactionExist(getWlptr());
return ret;
}
2024-04-10 12:27:10 +00:00
String getFilename() => monero.Wallet_filename(wptr!);
2024-04-12 12:54:24 +00:00
// TODO(mrcyjanek): Cake polyseed support
2024-04-10 12:27:10 +00:00
String getSeed() => monero.Wallet_seed(wptr!, seedOffset: '');
2024-04-12 12:54:24 +00:00
String getAddress({int accountIndex = 0, int addressIndex = 1}) => monero.Wallet_address(wptr!, accountIndex: accountIndex, addressIndex: addressIndex);
2024-04-10 12:27:10 +00:00
int getFullBalance({int accountIndex = 0}) => monero.Wallet_balance(wptr!, accountIndex: accountIndex);
2024-04-10 12:27:10 +00:00
int getUnlockedBalance({int accountIndex = 0}) => monero.Wallet_unlockedBalance(wptr!, accountIndex: accountIndex);
2024-04-10 12:27:10 +00:00
int getCurrentHeight() => monero.Wallet_blockChainHeight(wptr!);
2024-04-10 12:27:10 +00:00
int getNodeHeightSync() => monero.Wallet_daemonBlockChainHeight(wptr!);
2024-04-10 12:27:10 +00:00
bool isConnectedSync() => monero.Wallet_connected(wptr!) != 0;
bool setupNodeSync(
2022-10-12 17:09:57 +00:00
{required String address,
String? login,
String? password,
bool useSSL = false,
bool isLightWallet = false,
String? socksProxyAddress}) {
2024-04-12 12:54:24 +00:00
print('''
{
wptr!,
daemonAddress: $address,
useSsl: $useSSL,
proxyAddress: $socksProxyAddress ?? '',
daemonUsername: $login ?? '',
daemonPassword: $password ?? ''
}
''');
2024-04-10 12:27:10 +00:00
monero.Wallet_init(
wptr!,
daemonAddress: address,
useSsl: useSSL,
proxyAddress: socksProxyAddress ?? '',
daemonUsername: login ?? '',
daemonPassword: password ?? ''
);
// monero.Wallet_init3(wptr!, argv0: '', defaultLogBaseName: 'moneroc', console: true);
monero.Wallet_startRefresh(wptr!);
monero.Wallet_refreshAsync(wptr!);
final status = monero.Wallet_status(wptr!);
2024-04-12 12:54:24 +00:00
if (status != 0) {
final error = monero.Wallet_errorString(wptr!);
print("error: $error");
throw SetupWalletException(message: error);
}
2024-04-10 12:27:10 +00:00
return status == 0;
}
2024-04-10 12:27:10 +00:00
void startRefreshSync() {}
2024-04-10 12:27:10 +00:00
Future<bool> connectToNode() async {
return true;
}
2024-04-10 12:27:10 +00:00
void setRefreshFromBlockHeight({required int height}) => monero.Wallet_setRefreshFromBlockHeight(wptr!, refresh_from_block_height: height);
2024-04-10 12:27:10 +00:00
void setRecoveringFromSeed({required bool isRecovery}) => monero.Wallet_setRecoveringFromSeed(wptr!, recoveringFromSeed: isRecovery);
void storeSync() {
2024-04-10 12:27:10 +00:00
monero.Wallet_store(wptr!);
}
void setPasswordSync(String password) {
2024-04-10 12:27:10 +00:00
monero.Wallet_setPassword(wptr!, password: password);
2024-04-10 12:27:10 +00:00
final status = monero.Wallet_status(wptr!);
if (status == 0) {
throw Exception(monero.Wallet_errorString(wptr!));
}
}
2024-04-10 12:27:10 +00:00
void closeCurrentWallet() {
monero.Wallet_stop(wptr!);
}
2024-04-10 12:27:10 +00:00
String getSecretViewKey() => monero.Wallet_secretViewKey(wptr!);
2024-04-10 12:27:10 +00:00
String getPublicViewKey() => monero.Wallet_publicViewKey(wptr!);
2024-04-10 12:27:10 +00:00
String getSecretSpendKey() => monero.Wallet_secretSpendKey(wptr!);
2024-04-10 12:27:10 +00:00
String getPublicSpendKey() => monero.Wallet_publicSpendKey(wptr!);
class SyncListener {
2022-10-12 17:09:57 +00:00
SyncListener(this.onNewBlock, this.onNewTransaction)
: _cachedBlockchainHeight = 0,
_lastKnownBlockHeight = 0,
_initialSyncHeight = 0;
2022-10-12 17:09:57 +00:00
void Function(int, int, double) onNewBlock;
void Function() onNewTransaction;
2022-10-12 17:09:57 +00:00
Timer? _updateSyncInfoTimer;
int _cachedBlockchainHeight;
int _lastKnownBlockHeight;
int _initialSyncHeight;
Future<int> getNodeHeightOrUpdate(int baseHeight) async {
if (_cachedBlockchainHeight < baseHeight || _cachedBlockchainHeight == 0) {
_cachedBlockchainHeight = await getNodeHeight();
}
return _cachedBlockchainHeight;
}
void start() {
_cachedBlockchainHeight = 0;
_lastKnownBlockHeight = 0;
_initialSyncHeight = 0;
_updateSyncInfoTimer ??=
Timer.periodic(Duration(milliseconds: 1200), (_) async {
if (isNewTransactionExist()) {
2022-10-12 17:09:57 +00:00
onNewTransaction();
}
var syncHeight = getSyncingHeight();
if (syncHeight <= 0) {
syncHeight = getCurrentHeight();
}
if (_initialSyncHeight <= 0) {
_initialSyncHeight = syncHeight;
}
final bchHeight = await getNodeHeightOrUpdate(syncHeight);
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);
});
}
void stop() => _updateSyncInfoTimer?.cancel();
}
SyncListener setListeners(void Function(int, int, double) onNewBlock,
void Function() onNewTransaction) {
final listener = SyncListener(onNewBlock, onNewTransaction);
2024-04-10 12:27:10 +00:00
// setListenerNative();
return listener;
}
2024-04-10 12:27:10 +00:00
void onStartup() {}
void _storeSync(Object _) => storeSync();
2022-10-12 17:09:57 +00:00
bool _setupNodeSync(Map<String, Object?> args) {
final address = args['address'] as String;
final login = (args['login'] ?? '') as String;
final password = (args['password'] ?? '') as String;
final useSSL = args['useSSL'] as bool;
final isLightWallet = args['isLightWallet'] as bool;
final socksProxyAddress = (args['socksProxyAddress'] ?? '') as String;
return setupNodeSync(
address: address,
login: login,
password: password,
useSSL: useSSL,
isLightWallet: isLightWallet,
socksProxyAddress: socksProxyAddress);
}
bool _isConnected(Object _) => isConnectedSync();
int _getNodeHeight(Object _) => getNodeHeightSync();
void startRefresh() => startRefreshSync();
2022-10-12 17:09:57 +00:00
Future<void> setupNode(
{required String address,
String? login,
String? password,
bool useSSL = false,
String? socksProxyAddress,
2024-04-10 12:27:10 +00:00
bool isLightWallet = false}) async =>
_setupNodeSync({
'address': address,
2022-10-12 17:09:57 +00:00
'login': login ,
'password': password,
'useSSL': useSSL,
'isLightWallet': isLightWallet,
'socksProxyAddress': socksProxyAddress
});
2024-04-10 12:27:10 +00:00
Future<void> store() async => _storeSync(0);
2024-04-10 12:27:10 +00:00
Future<bool> isConnected() async => _isConnected(0);
2024-04-10 12:27:10 +00:00
Future<int> getNodeHeight() async => _getNodeHeight(0);
2024-04-10 12:27:10 +00:00
void rescanBlockchainAsync() => monero.Wallet_rescanBlockchainAsync(wptr!);
String getSubaddressLabel(int accountIndex, int addressIndex) {
2024-04-10 12:27:10 +00:00
return monero.Wallet_getSubaddressLabel(wptr!, accountIndex: accountIndex, addressIndex: addressIndex);
}
2024-04-10 12:27:10 +00:00
Future setTrustedDaemon(bool trusted) async => monero.Wallet_setTrustedDaemon(wptr!, arg: trusted);
2024-04-10 12:27:10 +00:00
Future<bool> trustedDaemon() async => monero.Wallet_trustedDaemon(wptr!);
String signMessage(String message, {String address = ""}) {
2024-04-10 12:27:10 +00:00
return monero.Wallet_signMessage(wptr!, message: message, address: address);
}