cake_wallet/cw_haven/lib/api/wallet.dart

400 lines
11 KiB
Dart
Raw Normal View History

2022-03-30 15:57:04 +00:00
import 'dart:async';
import 'dart:ffi';
import 'package:ffi/ffi.dart';
import 'package:cw_haven/api/structs/ut8_box.dart';
2022-03-30 15:57:04 +00:00
import 'package:cw_haven/api/convert_utf8_to_string.dart';
import 'package:cw_haven/api/signatures.dart';
import 'package:cw_haven/api/types.dart';
import 'package:cw_haven/api/haven_api.dart';
import 'package:cw_haven/api/exceptions/setup_wallet_exception.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/services.dart';
int _boolToInt(bool value) => value ? 1 : 0;
final getFileNameNative = havenApi
.lookup<NativeFunction<get_filename>>('get_filename')
.asFunction<GetFilename>();
final getSeedNative =
havenApi.lookup<NativeFunction<get_seed>>('seed').asFunction<GetSeed>();
final getAddressNative = havenApi
.lookup<NativeFunction<get_address>>('get_address')
.asFunction<GetAddress>();
final getFullBalanceNative = havenApi
.lookup<NativeFunction<get_full_balanace>>('get_full_balance')
.asFunction<GetFullBalance>();
final getUnlockedBalanceNative = havenApi
.lookup<NativeFunction<get_unlocked_balanace>>('get_unlocked_balance')
.asFunction<GetUnlockedBalance>();
final getCurrentHeightNative = havenApi
.lookup<NativeFunction<get_current_height>>('get_current_height')
.asFunction<GetCurrentHeight>();
final getNodeHeightNative = havenApi
.lookup<NativeFunction<get_node_height>>('get_node_height')
.asFunction<GetNodeHeight>();
final isConnectedNative = havenApi
.lookup<NativeFunction<is_connected>>('is_connected')
.asFunction<IsConnected>();
final setupNodeNative = havenApi
.lookup<NativeFunction<setup_node>>('setup_node')
.asFunction<SetupNode>();
final startRefreshNative = havenApi
.lookup<NativeFunction<start_refresh>>('start_refresh')
.asFunction<StartRefresh>();
final connecToNodeNative = havenApi
.lookup<NativeFunction<connect_to_node>>('connect_to_node')
.asFunction<ConnectToNode>();
final setRefreshFromBlockHeightNative = havenApi
.lookup<NativeFunction<set_refresh_from_block_height>>(
'set_refresh_from_block_height')
.asFunction<SetRefreshFromBlockHeight>();
final setRecoveringFromSeedNative = havenApi
.lookup<NativeFunction<set_recovering_from_seed>>(
'set_recovering_from_seed')
.asFunction<SetRecoveringFromSeed>();
final storeNative =
havenApi.lookup<NativeFunction<store_c>>('store').asFunction<Store>();
final setPasswordNative =
havenApi.lookup<NativeFunction<set_password>>('set_password').asFunction<SetPassword>();
2022-03-30 15:57:04 +00:00
final setListenerNative = havenApi
.lookup<NativeFunction<set_listener>>('set_listener')
.asFunction<SetListener>();
final getSyncingHeightNative = havenApi
.lookup<NativeFunction<get_syncing_height>>('get_syncing_height')
.asFunction<GetSyncingHeight>();
final isNeededToRefreshNative = havenApi
.lookup<NativeFunction<is_needed_to_refresh>>('is_needed_to_refresh')
.asFunction<IsNeededToRefresh>();
final isNewTransactionExistNative = havenApi
.lookup<NativeFunction<is_new_transaction_exist>>(
'is_new_transaction_exist')
.asFunction<IsNewTransactionExist>();
final getSecretViewKeyNative = havenApi
.lookup<NativeFunction<secret_view_key>>('secret_view_key')
.asFunction<SecretViewKey>();
final getPublicViewKeyNative = havenApi
.lookup<NativeFunction<public_view_key>>('public_view_key')
.asFunction<PublicViewKey>();
final getSecretSpendKeyNative = havenApi
.lookup<NativeFunction<secret_spend_key>>('secret_spend_key')
.asFunction<SecretSpendKey>();
final getPublicSpendKeyNative = havenApi
.lookup<NativeFunction<secret_view_key>>('public_spend_key')
.asFunction<PublicSpendKey>();
final closeCurrentWalletNative = havenApi
.lookup<NativeFunction<close_current_wallet>>('close_current_wallet')
.asFunction<CloseCurrentWallet>();
final onStartupNative = havenApi
.lookup<NativeFunction<on_startup>>('on_startup')
.asFunction<OnStartup>();
final rescanBlockchainAsyncNative = havenApi
.lookup<NativeFunction<rescan_blockchain>>('rescan_blockchain')
.asFunction<RescanBlockchainAsync>();
final setTrustedDaemonNative = havenApi
.lookup<NativeFunction<set_trusted_daemon>>('set_trusted_daemon')
.asFunction<SetTrustedDaemon>();
final trustedDaemonNative = havenApi
.lookup<NativeFunction<trusted_daemon>>('trusted_daemon')
.asFunction<TrustedDaemon>();
2023-10-02 14:17:35 +00:00
int getSyncingHeight() {
return getSyncingHeightNative();
}
2022-03-30 15:57:04 +00:00
bool isNeededToRefresh() => isNeededToRefreshNative() != 0;
2023-10-02 14:17:35 +00:00
bool isNewTransactionExist() {
return isNewTransactionExistNative() != 0;
}
2022-03-30 15:57:04 +00:00
String getFilename() => convertUTF8ToString(pointer: getFileNameNative());
String getSeed() => convertUTF8ToString(pointer: getSeedNative());
2023-10-02 14:17:35 +00:00
String getAddress({int accountIndex = 0, int addressIndex = 0}) {
return convertUTF8ToString(pointer: getAddressNative(accountIndex, addressIndex));
}
2022-03-30 15:57:04 +00:00
int getFullBalance({int accountIndex = 0}) =>
getFullBalanceNative(accountIndex);
int getUnlockedBalance({int accountIndex = 0}) =>
getUnlockedBalanceNative(accountIndex);
2023-10-02 14:17:35 +00:00
int getCurrentHeight() {
return getCurrentHeightNative();
}
2022-03-30 15:57:04 +00:00
2023-10-02 14:17:35 +00:00
int getNodeHeightSync() {
return getNodeHeightNative();
}
2022-03-30 15:57:04 +00:00
2023-10-02 14:17:35 +00:00
bool isConnectedSync() {
return isConnectedNative() != 0;
}
2022-03-30 15:57:04 +00:00
bool setupNodeSync(
2022-10-12 17:09:57 +00:00
{required String address,
String? login,
String? password,
2022-03-30 15:57:04 +00:00
bool useSSL = false,
bool isLightWallet = false,
String? socksProxyAddress}) {
2022-10-12 17:09:57 +00:00
final addressPointer = address.toNativeUtf8();
Pointer<Utf8>? loginPointer;
Pointer<Utf8>? socksProxyAddressPointer;
2022-10-12 17:09:57 +00:00
Pointer<Utf8>? passwordPointer;
2022-03-30 15:57:04 +00:00
if (login != null) {
2022-10-12 17:09:57 +00:00
loginPointer = login.toNativeUtf8();
2022-03-30 15:57:04 +00:00
}
if (password != null) {
2022-10-12 17:09:57 +00:00
passwordPointer = password.toNativeUtf8();
2022-03-30 15:57:04 +00:00
}
if (socksProxyAddress != null) {
socksProxyAddressPointer = socksProxyAddress.toNativeUtf8();
}
2022-10-12 17:09:57 +00:00
final errorMessagePointer = ''.toNativeUtf8();
2022-03-30 15:57:04 +00:00
final isSetupNode = setupNodeNative(
addressPointer,
loginPointer,
passwordPointer,
_boolToInt(useSSL),
_boolToInt(isLightWallet),
socksProxyAddressPointer,
2022-03-30 15:57:04 +00:00
errorMessagePointer) !=
0;
2022-10-12 17:09:57 +00:00
calloc.free(addressPointer);
if (loginPointer != null) {
calloc.free(loginPointer);
}
if (passwordPointer != null) {
calloc.free(passwordPointer);
}
2022-03-30 15:57:04 +00:00
if (!isSetupNode) {
throw SetupWalletException(
message: convertUTF8ToString(pointer: errorMessagePointer));
}
return isSetupNode;
}
void startRefreshSync() => startRefreshNative();
Future<bool> connectToNode() async => connecToNodeNative() != 0;
2023-10-02 14:17:35 +00:00
void setRefreshFromBlockHeight({required int height}) {
setRefreshFromBlockHeightNative(height);
}
2022-03-30 15:57:04 +00:00
2023-10-02 14:17:35 +00:00
void setRecoveringFromSeed({required bool isRecovery}) {
setRecoveringFromSeedNative(_boolToInt(isRecovery));
}
2022-03-30 15:57:04 +00:00
void storeSync() {
2022-10-12 17:09:57 +00:00
final pathPointer = ''.toNativeUtf8();
2022-03-30 15:57:04 +00:00
storeNative(pathPointer);
2022-10-12 17:09:57 +00:00
calloc.free(pathPointer);
2022-03-30 15:57:04 +00:00
}
void setPasswordSync(String password) {
2022-10-12 17:09:57 +00:00
final passwordPointer = password.toNativeUtf8();
final errorMessagePointer = calloc<Utf8Box>();
final changed = setPasswordNative(passwordPointer, errorMessagePointer) != 0;
2022-10-12 17:09:57 +00:00
calloc.free(passwordPointer);
if (!changed) {
final message = errorMessagePointer.ref.getValue();
2022-10-12 17:09:57 +00:00
calloc.free(errorMessagePointer);
throw Exception(message);
}
2022-10-12 17:09:57 +00:00
calloc.free(errorMessagePointer);
}
2022-03-30 15:57:04 +00:00
void closeCurrentWallet() => closeCurrentWalletNative();
2023-10-02 14:17:35 +00:00
String getSecretViewKey() {
return convertUTF8ToString(pointer: getSecretViewKeyNative());
}
2022-03-30 15:57:04 +00:00
2023-10-02 14:17:35 +00:00
String getPublicViewKey() {
return convertUTF8ToString(pointer: getPublicViewKeyNative());
}
2022-03-30 15:57:04 +00:00
2023-10-02 14:17:35 +00:00
String getSecretSpendKey() {
return convertUTF8ToString(pointer: getSecretSpendKeyNative());
}
2022-03-30 15:57:04 +00:00
2023-10-02 14:17:35 +00:00
String getPublicSpendKey() {
return convertUTF8ToString(pointer: getPublicSpendKeyNative());
}
2022-03-30 15:57:04 +00:00
class SyncListener {
2022-10-12 17:09:57 +00:00
SyncListener(this.onNewBlock, this.onNewTransaction)
: _cachedBlockchainHeight = 0,
_lastKnownBlockHeight = 0,
_initialSyncHeight = 0;
2022-03-30 15:57:04 +00:00
void Function(int, int, double) onNewBlock;
void Function() onNewTransaction;
2022-10-12 17:09:57 +00:00
Timer? _updateSyncInfoTimer;
2022-03-30 15:57:04 +00:00
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()) {
onNewTransaction?.call();
}
var syncHeight = getSyncingHeight();
if (syncHeight <= 0) {
syncHeight = getCurrentHeight();
}
if (_initialSyncHeight <= 0) {
_initialSyncHeight = syncHeight;
}
final bchHeight = await getNodeHeightOrUpdate(syncHeight);
if (_lastKnownBlockHeight == syncHeight || syncHeight == null) {
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);
setListenerNative();
return listener;
}
2023-10-02 14:17:35 +00:00
void onStartup() {
onStartupNative();
}
2022-03-30 15:57:04 +00:00
void _storeSync(Object _) => storeSync();
bool _setupNodeSync(Map 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;
2022-03-30 15:57:04 +00:00
return setupNodeSync(
address: address,
login: login,
password: password,
useSSL: useSSL,
isLightWallet: isLightWallet,
socksProxyAddress: socksProxyAddress);
2022-03-30 15:57:04 +00:00
}
bool _isConnected(Object _) => isConnectedSync();
int _getNodeHeight(Object _) => getNodeHeightSync();
2023-10-02 14:17:35 +00:00
void startRefresh() {
startRefreshSync();
}
2022-03-30 15:57:04 +00:00
2022-10-12 17:09:57 +00:00
Future<void> setupNode(
{required String address,
String? login,
String? password,
2022-03-30 15:57:04 +00:00
bool useSSL = false,
String? socksProxyAddress,
2022-03-30 15:57:04 +00:00
bool isLightWallet = false}) =>
2022-10-12 17:09:57 +00:00
compute<Map<String, Object?>, void>(_setupNodeSync, {
2022-03-30 15:57:04 +00:00
'address': address,
'login': login,
'password': password,
'useSSL': useSSL,
'isLightWallet': isLightWallet,
'socksProxyAddress': socksProxyAddress
2022-03-30 15:57:04 +00:00
});
2022-10-12 17:09:57 +00:00
Future<void> store() => compute<int, void>(_storeSync, 0);
2022-03-30 15:57:04 +00:00
Future<bool> isConnected() => compute(_isConnected, 0);
Future<int> getNodeHeight() => compute(_getNodeHeight, 0);
2023-10-02 14:17:35 +00:00
void rescanBlockchainAsync() {
rescanBlockchainAsyncNative();
}
Future setTrustedDaemon(bool trusted) async => setTrustedDaemonNative(_boolToInt(trusted));
Future<bool> trustedDaemon() async => trustedDaemonNative() != 0;