WIP: paginated spark anon set prep

This commit is contained in:
julian 2024-12-14 15:29:17 -06:00 committed by julian-CStack
parent 744f273862
commit d6aec00b58
10 changed files with 420 additions and 69 deletions

View file

@ -9,6 +9,7 @@ import 'package:sqlite3/sqlite3.dart';
import 'package:uuid/uuid.dart'; import 'package:uuid/uuid.dart';
import '../../electrumx_rpc/electrumx_client.dart'; import '../../electrumx_rpc/electrumx_client.dart';
import '../../models/electrumx_response/spark_models.dart';
import '../../utilities/extensions/extensions.dart'; import '../../utilities/extensions/extensions.dart';
import '../../utilities/logger.dart'; import '../../utilities/logger.dart';
import '../../utilities/stack_file_system.dart'; import '../../utilities/stack_file_system.dart';
@ -42,12 +43,17 @@ abstract class _FiroCache {
network == CryptoCurrencyNetwork.main network == CryptoCurrencyNetwork.main
? "spark_set_v$_setCacheVersion.sqlite3" ? "spark_set_v$_setCacheVersion.sqlite3"
: "spark_set_v${_setCacheVersion}_${network.name}.sqlite3"; : "spark_set_v${_setCacheVersion}_${network.name}.sqlite3";
static String sparkSetMetaCacheFileName(CryptoCurrencyNetwork network) =>
network == CryptoCurrencyNetwork.main
? "spark_set_meta_v$_setCacheVersion.sqlite3"
: "spark_set_meta_v${_setCacheVersion}_${network.name}.sqlite3";
static String sparkUsedTagsCacheFileName(CryptoCurrencyNetwork network) => static String sparkUsedTagsCacheFileName(CryptoCurrencyNetwork network) =>
network == CryptoCurrencyNetwork.main network == CryptoCurrencyNetwork.main
? "spark_tags_v$_tagsCacheVersion.sqlite3" ? "spark_tags_v$_tagsCacheVersion.sqlite3"
: "spark_tags_v${_tagsCacheVersion}_${network.name}.sqlite3"; : "spark_tags_v${_tagsCacheVersion}_${network.name}.sqlite3";
static final Map<CryptoCurrencyNetwork, Database> _setCacheDB = {}; static final Map<CryptoCurrencyNetwork, Database> _setCacheDB = {};
static final Map<CryptoCurrencyNetwork, Database> _setMetaCacheDB = {};
static final Map<CryptoCurrencyNetwork, Database> _usedTagsCacheDB = {}; static final Map<CryptoCurrencyNetwork, Database> _usedTagsCacheDB = {};
static Database setCacheDB(CryptoCurrencyNetwork network) { static Database setCacheDB(CryptoCurrencyNetwork network) {
if (_setCacheDB[network] == null) { if (_setCacheDB[network] == null) {
@ -58,6 +64,15 @@ abstract class _FiroCache {
return _setCacheDB[network]!; return _setCacheDB[network]!;
} }
static Database setMetaCacheDB(CryptoCurrencyNetwork network) {
if (_setMetaCacheDB[network] == null) {
throw Exception(
"FiroCache.init() must be called before accessing FiroCache.db!",
);
}
return _setMetaCacheDB[network]!;
}
static Database usedTagsCacheDB(CryptoCurrencyNetwork network) { static Database usedTagsCacheDB(CryptoCurrencyNetwork network) {
if (_usedTagsCacheDB[network] == null) { if (_usedTagsCacheDB[network] == null) {
throw Exception( throw Exception(
@ -78,12 +93,18 @@ abstract class _FiroCache {
final sparkSetCacheFile = final sparkSetCacheFile =
File("${sqliteDir.path}/${sparkSetCacheFileName(network)}"); File("${sqliteDir.path}/${sparkSetCacheFileName(network)}");
final sparkSetMetaCacheFile =
File("${sqliteDir.path}/${sparkSetMetaCacheFileName(network)}");
final sparkUsedTagsCacheFile = final sparkUsedTagsCacheFile =
File("${sqliteDir.path}/${sparkUsedTagsCacheFileName(network)}"); File("${sqliteDir.path}/${sparkUsedTagsCacheFileName(network)}");
if (!(await sparkSetCacheFile.exists())) { if (!(await sparkSetCacheFile.exists())) {
await _createSparkSetCacheDb(sparkSetCacheFile.path); await _createSparkSetCacheDb(sparkSetCacheFile.path);
} }
if (!(await sparkSetMetaCacheFile.exists())) {
await _createSparkSetMetaCacheDb(sparkSetMetaCacheFile.path);
}
if (!(await sparkUsedTagsCacheFile.exists())) { if (!(await sparkUsedTagsCacheFile.exists())) {
await _createSparkUsedTagsCacheDb(sparkUsedTagsCacheFile.path); await _createSparkUsedTagsCacheDb(sparkUsedTagsCacheFile.path);
} }
@ -92,6 +113,10 @@ abstract class _FiroCache {
sparkSetCacheFile.path, sparkSetCacheFile.path,
mode: OpenMode.readWrite, mode: OpenMode.readWrite,
); );
_setMetaCacheDB[network] = sqlite3.open(
sparkSetMetaCacheFile.path,
mode: OpenMode.readWrite,
);
_usedTagsCacheDB[network] = sqlite3.open( _usedTagsCacheDB[network] = sqlite3.open(
sparkUsedTagsCacheFile.path, sparkUsedTagsCacheFile.path,
mode: OpenMode.readWrite, mode: OpenMode.readWrite,
@ -109,6 +134,12 @@ abstract class _FiroCache {
VACUUM; VACUUM;
""", """,
); );
setMetaCacheDB(network).execute(
"""
DELETE FROM PreviousMetaFetchResult;
VACUUM;
""",
);
usedTagsCacheDB(network).execute( usedTagsCacheDB(network).execute(
""" """
DELETE FROM SparkUsedCoinTags; DELETE FROM SparkUsedCoinTags;
@ -159,6 +190,27 @@ abstract class _FiroCache {
db.dispose(); db.dispose();
} }
static Future<void> _createSparkSetMetaCacheDb(String file) async {
final db = sqlite3.open(
file,
mode: OpenMode.readWriteCreate,
);
db.execute(
"""
CREATE TABLE PreviousMetaFetchResult (
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT UNIQUE,
coinGroupId INTEGER NOT NULL UNIQUE,
blockHash TEXT NOT NULL,
setHash TEXT NOT NULL,
size INTEGER NOT NULL
);
""",
);
db.dispose();
}
static Future<void> _createSparkUsedTagsCacheDb(String file) async { static Future<void> _createSparkUsedTagsCacheDb(String file) async {
final db = sqlite3.open( final db = sqlite3.open(
file, file,

View file

@ -6,6 +6,8 @@ typedef LTagPair = ({String tag, String txid});
/// background isolate and [FiroCacheCoordinator] should manage that isolate /// background isolate and [FiroCacheCoordinator] should manage that isolate
abstract class FiroCacheCoordinator { abstract class FiroCacheCoordinator {
static final Map<CryptoCurrencyNetwork, _FiroCacheWorker> _workers = {}; static final Map<CryptoCurrencyNetwork, _FiroCacheWorker> _workers = {};
static final Map<CryptoCurrencyNetwork, Mutex> _tagLocks = {};
static final Map<CryptoCurrencyNetwork, Mutex> _setLocks = {};
static bool _init = false; static bool _init = false;
static Future<void> init() async { static Future<void> init() async {
@ -15,6 +17,8 @@ abstract class FiroCacheCoordinator {
_init = true; _init = true;
await _FiroCache.init(); await _FiroCache.init();
for (final network in _FiroCache.networks) { for (final network in _FiroCache.networks) {
_tagLocks[network] = Mutex();
_setLocks[network] = Mutex();
_workers[network] = await _FiroCacheWorker.spawn(network); _workers[network] = await _FiroCacheWorker.spawn(network);
} }
} }
@ -28,14 +32,26 @@ abstract class FiroCacheCoordinator {
final setCacheFile = File( final setCacheFile = File(
"${dir.path}/${_FiroCache.sparkSetCacheFileName(network)}", "${dir.path}/${_FiroCache.sparkSetCacheFileName(network)}",
); );
final setMetaCacheFile = File(
"${dir.path}/${_FiroCache.sparkSetMetaCacheFileName(network)}",
);
final usedTagsCacheFile = File( final usedTagsCacheFile = File(
"${dir.path}/${_FiroCache.sparkUsedTagsCacheFileName(network)}", "${dir.path}/${_FiroCache.sparkUsedTagsCacheFileName(network)}",
); );
final int bytes =
((await setCacheFile.exists()) ? await setCacheFile.length() : 0) + final setSize =
((await usedTagsCacheFile.exists()) (await setCacheFile.exists()) ? await setCacheFile.length() : 0;
final tagsSize = (await usedTagsCacheFile.exists())
? await usedTagsCacheFile.length() ? await usedTagsCacheFile.length()
: 0); : 0;
final setMetaSize =
(await setMetaCacheFile.exists()) ? await setMetaCacheFile.length() : 0;
print("TAG SIZE: $tagsSize");
print("SET SIZE: $setSize");
print("SET META SIZE: $setMetaSize");
final int bytes = tagsSize + setSize + setMetaSize;
if (bytes < 1024) { if (bytes < 1024) {
return '$bytes B'; return '$bytes B';
@ -55,8 +71,10 @@ abstract class FiroCacheCoordinator {
ElectrumXClient client, ElectrumXClient client,
CryptoCurrencyNetwork network, CryptoCurrencyNetwork network,
) async { ) async {
await _tagLocks[network]!.protect(() async {
final count = await FiroCacheCoordinator.getUsedCoinTagsCount(network); final count = await FiroCacheCoordinator.getUsedCoinTagsCount(network);
final unhashedTags = await client.getSparkUnhashedUsedCoinsTagsWithTxHashes( final unhashedTags =
await client.getSparkUnhashedUsedCoinsTagsWithTxHashes(
startNumber: count, startNumber: count,
); );
if (unhashedTags.isNotEmpty) { if (unhashedTags.isNotEmpty) {
@ -67,13 +85,21 @@ abstract class FiroCacheCoordinator {
), ),
); );
} }
});
} }
static Future<void> runFetchAndUpdateSparkAnonSetCacheForGroupId( static Future<void> runFetchAndUpdateSparkAnonSetCacheForGroupId(
int groupId, int groupId,
ElectrumXClient client, ElectrumXClient client,
CryptoCurrencyNetwork network, CryptoCurrencyNetwork network,
void Function(int countFetched, int totalCount)? progressUpdated,
) async { ) async {
await _setLocks[network]!.protect(() async {
Map<String, dynamic> json;
SparkAnonymitySetMeta? meta;
if (progressUpdated == null) {
// Legacy
final blockhashResult = final blockhashResult =
await FiroCacheCoordinator.getLatestSetInfoForGroupId( await FiroCacheCoordinator.getLatestSetInfoForGroupId(
groupId, groupId,
@ -81,10 +107,63 @@ abstract class FiroCacheCoordinator {
); );
final blockHash = blockhashResult?.blockHash ?? ""; final blockHash = blockhashResult?.blockHash ?? "";
final json = await client.getSparkAnonymitySet( json = await client.getSparkAnonymitySet(
coinGroupId: groupId.toString(), coinGroupId: groupId.toString(),
startBlockHash: blockHash.toHexReversedFromBase64, startBlockHash: blockHash.toHexReversedFromBase64,
); );
} else {
const sectorSize = 100; // TODO adjust this?
final prevMetaSize =
await FiroCacheCoordinator.getSparkMetaSetSizeForGroupId(
groupId,
network,
);
final prevSize = prevMetaSize ?? 0;
meta = await client.getSparkAnonymitySetMeta(
coinGroupId: groupId,
);
progressUpdated.call(prevSize, meta.size);
/// Returns blockHash (last block hash),
/// setHash (hash of current set)
/// and coins (the list of pairs serialized coin and tx hash)
final fullSectorCount = (meta.size - prevSize) ~/ sectorSize;
final remainder = (meta.size - prevSize) % sectorSize;
final List<dynamic> coins = [];
for (int i = 0; i < fullSectorCount; i++) {
final start = (i * sectorSize) + prevSize;
final data = await client.getSparkAnonymitySetBySector(
coinGroupId: groupId,
latestBlock: meta.blockHash,
startIndex: start,
endIndex: start + sectorSize,
);
coins.addAll(data);
}
if (remainder > 0) {
final data = await client.getSparkAnonymitySetBySector(
coinGroupId: groupId,
latestBlock: meta.blockHash,
startIndex: meta.size - remainder,
endIndex: meta.size,
);
coins.addAll(data);
}
json = {
"blockHash": meta.blockHash,
"setHash": meta.setHash,
"coins": coins,
};
}
await _workers[network]!.runTask( await _workers[network]!.runTask(
FCTask( FCTask(
@ -92,6 +171,16 @@ abstract class FiroCacheCoordinator {
data: (groupId, json), data: (groupId, json),
), ),
); );
if (meta != null) {
await _workers[network]!.runTask(
FCTask(
func: FCFuncName._updateSparkAnonSetMetaWith,
data: meta,
),
);
}
});
} }
// =========================================================================== // ===========================================================================
@ -228,4 +317,19 @@ abstract class FiroCacheCoordinator {
db: _FiroCache.setCacheDB(network), db: _FiroCache.setCacheDB(network),
); );
} }
static Future<int?> getSparkMetaSetSizeForGroupId(
int groupId,
CryptoCurrencyNetwork network,
) async {
final result = await _Reader._getSizeForGroupId(
groupId,
db: _FiroCache.setMetaCacheDB(network),
);
if (result.isEmpty) {
return null;
}
return result.first["size"] as int;
}
} }

View file

@ -56,6 +56,21 @@ abstract class _Reader {
return db.select("$query;").first["setExists"] == 1; return db.select("$query;").first["setExists"] == 1;
} }
// ===========================================================================
// =============== Spark anonymity set meta queries ==========================
static Future<ResultSet> _getSizeForGroupId(
int groupId, {
required Database db,
}) async {
final query = """
SELECT ss.size
FROM PreviousMetaFetchResult ss
WHERE ss.groupId = $groupId;
""";
return db.select("$query;");
}
// =========================================================================== // ===========================================================================
// =============== Spark used coin tags queries ============================== // =============== Spark used coin tags queries ==============================

View file

@ -3,6 +3,7 @@ part of 'firo_cache.dart';
enum FCFuncName { enum FCFuncName {
_updateSparkAnonSetCoinsWith, _updateSparkAnonSetCoinsWith,
_updateSparkUsedTagsWith, _updateSparkUsedTagsWith,
_updateSparkAnonSetMetaWith,
} }
class FCTask { class FCTask {
@ -29,6 +30,8 @@ class _FiroCacheWorker {
final dir = await StackFileSystem.applicationFiroCacheSQLiteDirectory(); final dir = await StackFileSystem.applicationFiroCacheSQLiteDirectory();
final setCacheFilePath = final setCacheFilePath =
"${dir.path}/${_FiroCache.sparkSetCacheFileName(network)}"; "${dir.path}/${_FiroCache.sparkSetCacheFileName(network)}";
final setMetaCacheFilePath =
"${dir.path}/${_FiroCache.sparkSetMetaCacheFileName(network)}";
final usedTagsCacheFilePath = final usedTagsCacheFilePath =
"${dir.path}/${_FiroCache.sparkUsedTagsCacheFileName(network)}"; "${dir.path}/${_FiroCache.sparkUsedTagsCacheFileName(network)}";
@ -48,7 +51,12 @@ class _FiroCacheWorker {
try { try {
await Isolate.spawn( await Isolate.spawn(
_startWorkerIsolate, _startWorkerIsolate,
(initPort.sendPort, setCacheFilePath, usedTagsCacheFilePath), (
initPort.sendPort,
setCacheFilePath,
setMetaCacheFilePath,
usedTagsCacheFilePath,
),
); );
} catch (_) { } catch (_) {
initPort.close(); initPort.close();
@ -79,6 +87,7 @@ class _FiroCacheWorker {
ReceivePort receivePort, ReceivePort receivePort,
SendPort sendPort, SendPort sendPort,
Database setCacheDb, Database setCacheDb,
Database setMetaCacheDb,
Database usedTagsCacheDb, Database usedTagsCacheDb,
Mutex mutex, Mutex mutex,
) { ) {
@ -104,6 +113,13 @@ class _FiroCacheWorker {
task.data as List<List<dynamic>>, task.data as List<List<dynamic>>,
); );
break; break;
case FCFuncName._updateSparkAnonSetMetaWith:
result = _updateSparkAnonSetMetaWith(
setMetaCacheDb,
task.data as SparkAnonymitySetMeta,
);
break;
} }
if (result.success) { if (result.success) {
@ -118,7 +134,7 @@ class _FiroCacheWorker {
}); });
} }
static void _startWorkerIsolate((SendPort, String, String) args) { static void _startWorkerIsolate((SendPort, String, String, String) args) {
final receivePort = ReceivePort(); final receivePort = ReceivePort();
args.$1.send(receivePort.sendPort); args.$1.send(receivePort.sendPort);
final mutex = Mutex(); final mutex = Mutex();
@ -126,14 +142,19 @@ class _FiroCacheWorker {
args.$2, args.$2,
mode: OpenMode.readWrite, mode: OpenMode.readWrite,
); );
final usedTagsCacheDb = sqlite3.open( final setMetaCacheDb = sqlite3.open(
args.$3, args.$3,
mode: OpenMode.readWrite, mode: OpenMode.readWrite,
); );
final usedTagsCacheDb = sqlite3.open(
args.$4,
mode: OpenMode.readWrite,
);
_handleCommandsToIsolate( _handleCommandsToIsolate(
receivePort, receivePort,
args.$1, args.$1,
setCacheDb, setCacheDb,
setMetaCacheDb,
usedTagsCacheDb, usedTagsCacheDb,
mutex, mutex,
); );

View file

@ -48,6 +48,31 @@ FCResult _updateSparkUsedTagsWith(
} }
} }
// ===========================================================================
// ================== write to spark anon set Meta cache ==========================
FCResult _updateSparkAnonSetMetaWith(
Database db,
SparkAnonymitySetMeta meta,
) {
db.execute("BEGIN;");
try {
db.execute(
"""
INSERT OR REPLACE INTO PreviousMetaFetchResult (coinGroupId, blockHash, setHash, size)
VALUES (?, ?, ?, ?);
""",
[meta.coinGroupId, meta.blockHash, meta.setHash, meta.size],
);
db.execute("COMMIT;");
return FCResult(success: true);
} catch (e) {
db.execute("ROLLBACK;");
return FCResult(success: false, error: e);
}
}
// =========================================================================== // ===========================================================================
// ================== write to spark anon set cache ========================== // ================== write to spark anon set cache ==========================

View file

@ -21,6 +21,7 @@ import 'package:mutex/mutex.dart';
import 'package:stream_channel/stream_channel.dart'; import 'package:stream_channel/stream_channel.dart';
import '../exceptions/electrumx/no_such_transaction.dart'; import '../exceptions/electrumx/no_such_transaction.dart';
import '../models/electrumx_response/spark_models.dart';
import '../services/event_bus/events/global/tor_connection_status_changed_event.dart'; import '../services/event_bus/events/global/tor_connection_status_changed_event.dart';
import '../services/event_bus/events/global/tor_status_changed_event.dart'; import '../services/event_bus/events/global/tor_status_changed_event.dart';
import '../services/event_bus/global_event_bus.dart'; import '../services/event_bus/global_event_bus.dart';
@ -34,13 +35,6 @@ import '../wallets/crypto_currency/crypto_currency.dart';
import '../wallets/crypto_currency/interfaces/electrumx_currency_interface.dart'; import '../wallets/crypto_currency/interfaces/electrumx_currency_interface.dart';
import 'client_manager.dart'; import 'client_manager.dart';
typedef SparkMempoolData = ({
String txid,
List<String> serialContext,
List<String> lTags,
List<String> coins,
});
class WifiOnlyException implements Exception {} class WifiOnlyException implements Exception {}
class TorOnlyException implements Exception {} class TorOnlyException implements Exception {}
@ -1037,29 +1031,30 @@ class ElectrumXClient {
/// "b476ed2b374bb081ea51d111f68f0136252521214e213d119b8dc67b92f5a390", /// "b476ed2b374bb081ea51d111f68f0136252521214e213d119b8dc67b92f5a390",
/// ] /// ]
/// } /// }
Future<List<Map<String, dynamic>>> getSparkMintMetaData({ /// NOT USED?
String? requestID, // Future<List<Map<String, dynamic>>> getSparkMintMetaData({
required List<String> sparkCoinHashes, // String? requestID,
}) async { // required List<String> sparkCoinHashes,
try { // }) async {
Logging.instance.log( // try {
"attempting to fetch spark.getsparkmintmetadata...", // Logging.instance.log(
level: LogLevel.Info, // "attempting to fetch spark.getsparkmintmetadata...",
); // level: LogLevel.Info,
await checkElectrumAdapter(); // );
final List<dynamic> response = // await checkElectrumAdapter();
await (getElectrumAdapter() as FiroElectrumClient) // final List<dynamic> response =
.getSparkMintMetaData(sparkCoinHashes: sparkCoinHashes); // await (getElectrumAdapter() as FiroElectrumClient)
Logging.instance.log( // .getSparkMintMetaData(sparkCoinHashes: sparkCoinHashes);
"Fetching spark.getsparkmintmetadata finished", // Logging.instance.log(
level: LogLevel.Info, // "Fetching spark.getsparkmintmetadata finished",
); // level: LogLevel.Info,
return List<Map<String, dynamic>>.from(response); // );
} catch (e) { // return List<Map<String, dynamic>>.from(response);
Logging.instance.log(e, level: LogLevel.Error); // } catch (e) {
rethrow; // Logging.instance.log(e, level: LogLevel.Error);
} // rethrow;
} // }
// }
/// Returns the latest Spark set id /// Returns the latest Spark set id
/// ///
@ -1135,7 +1130,7 @@ class ElectrumXClient {
final List<SparkMempoolData> result = []; final List<SparkMempoolData> result = [];
for (final entry in map.entries) { for (final entry in map.entries) {
result.add( result.add(
( SparkMempoolData(
txid: entry.key, txid: entry.key,
serialContext: serialContext:
List<String>.from(entry.value["serial_context"] as List), List<String>.from(entry.value["serial_context"] as List),
@ -1191,6 +1186,94 @@ class ElectrumXClient {
rethrow; rethrow;
} }
} }
// ======== New Paginated Endpoints ==========================================
Future<SparkAnonymitySetMeta> getSparkAnonymitySetMeta({
String? requestID,
required int coinGroupId,
}) async {
try {
const command =
"spark.getsparkanonyumitysetmeta"; // TODO verify this will be correct
final start = DateTime.now();
final response = await request(
requestID: requestID,
command: command,
args: [
"$coinGroupId",
],
);
final map = Map<String, dynamic>.from(response as Map);
final result = SparkAnonymitySetMeta(
coinGroupId: coinGroupId,
blockHash: map["blockHash"] as String,
setHash: map["setHash"] as String,
size: map["size"] as int,
);
Logging.instance.log(
"Finished ElectrumXClient.getSparkAnonymitySetMeta("
"requestID=$requestID, "
"coinGroupId=$coinGroupId"
"). Set meta=$result, "
"Duration=${DateTime.now().difference(start)}",
level: LogLevel.Debug,
);
return result;
} catch (e) {
Logging.instance.log(e, level: LogLevel.Error);
rethrow;
}
}
Future<List<dynamic>> getSparkAnonymitySetBySector({
String? requestID,
required int coinGroupId,
required String latestBlock,
required int startIndex, // inclusive
required int endIndex, // exclusive
}) async {
try {
const command =
"spark.getsparkanonyumitysetsector"; // TODO verify this will be correct
final start = DateTime.now();
final response = await request(
requestID: requestID,
command: command,
args: [
"$coinGroupId",
latestBlock,
"$startIndex",
"$endIndex",
],
);
final map = Map<String, dynamic>.from(response as Map);
final result = map["coins"] as List;
Logging.instance.log(
"Finished ElectrumXClient.getSparkAnonymitySetBySector("
"requestID=$requestID, "
"coinGroupId=$coinGroupId, "
"latestBlock=$latestBlock, "
"startIndex=$startIndex, "
"endIndex=$endIndex"
"). # of coins=${result.length}, "
"Duration=${DateTime.now().difference(start)}",
level: LogLevel.Debug,
);
return result;
} catch (e) {
Logging.instance.log(e, level: LogLevel.Error);
rethrow;
}
}
// =========================================================================== // ===========================================================================
Future<bool> isMasterNodeCollateral({ Future<bool> isMasterNodeCollateral({

View file

@ -0,0 +1,47 @@
class SparkMempoolData {
final String txid;
final List<String> serialContext;
final List<String> lTags;
final List<String> coins;
SparkMempoolData({
required this.txid,
required this.serialContext,
required this.lTags,
required this.coins,
});
@override
String toString() {
return "SparkMempoolData{"
"txid: $txid, "
"serialContext: $serialContext, "
"lTags: $lTags, "
"coins: $coins"
"}";
}
}
class SparkAnonymitySetMeta {
final int coinGroupId;
final String blockHash;
final String setHash;
final int size;
SparkAnonymitySetMeta({
required this.coinGroupId,
required this.blockHash,
required this.setHash,
required this.size,
});
@override
String toString() {
return "SparkAnonymitySetMeta{"
"coinGroupId: $coinGroupId, "
"blockHash: $blockHash, "
"setHash: $setHash, "
"size: $size"
"}";
}
}

View file

@ -725,6 +725,7 @@ class FiroWallet<T extends ElectrumXCurrencyInterface> extends Bip39HDWallet<T>
i, i,
electrumXClient, electrumXClient,
cryptoCurrency.network, cryptoCurrency.network,
null,
), ),
); );
} }

View file

@ -670,7 +670,7 @@ abstract class Wallet<T extends CryptoCurrency> {
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.3, walletId)); GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.3, walletId));
if (this is SparkInterface && !viewOnly) { if (this is SparkInterface && !viewOnly) {
// this should be called before updateTransactions() // this should be called before updateTransactions()
await (this as SparkInterface).refreshSparkData(); await (this as SparkInterface).refreshSparkData(null);
} }
_checkAlive(); _checkAlive();

View file

@ -795,7 +795,9 @@ mixin SparkInterface<T extends ElectrumXCurrencyInterface>
} }
} }
Future<void> refreshSparkData() async { Future<void> refreshSparkData(
void Function(int countFetched, int totalCount)? progressUpdated,
) async {
final start = DateTime.now(); final start = DateTime.now();
try { try {
// start by checking if any previous sets are missing from db and add the // start by checking if any previous sets are missing from db and add the
@ -823,6 +825,7 @@ mixin SparkInterface<T extends ElectrumXCurrencyInterface>
e, e,
electrumXClient, electrumXClient,
cryptoCurrency.network, cryptoCurrency.network,
null,
), ),
); );
@ -1086,7 +1089,7 @@ mixin SparkInterface<T extends ElectrumXCurrencyInterface>
} }
try { try {
await refreshSparkData(); await refreshSparkData(null);
} catch (e, s) { } catch (e, s) {
Logging.instance.log( Logging.instance.log(
"$runtimeType $walletId ${info.name}: $e\n$s", "$runtimeType $walletId ${info.name}: $e\n$s",